手机版
你好,游客 登录 注册
背景:
阅读新闻

C++随机排序容器中的元素

[日期:2019-02-04] 来源:cnblogs.com/apocelipes  作者:apocelipes [字体: ]

在各种程序语言中都提供了将容器元素随机排序的shuffle方法,C++也不例外。

不过C++将shuffle放在了<algorithm>中而不是像其他语言一样在random里,同时C++17删除了原先的random_shuffle新的程序应该使用c++11添加进去的std::shuffle。其中一个好处是新的函数在可以自定义随机数生成方法的同时保证了更好的安全性。

先来看下新函数的原型:
template< class RandomIt, class URBG >
void shuffle( RandomIt first, RandomIt last, URBG&& g );

其中first和last指定需要随机排序的范围,g是一个“UniformRandomBitGenerator”,就是一个可以产生规定范围内的随机数的可调用对象。

所以g可以是std::random_device或者像std::default_random_engine这样的随机数引擎,也可以是std::mt19937这样的标准库提供的随机数生成器的对象,它们都在<random>中。

shuffle调用后目标容器内的元素排列顺序会被随机打乱,我们看个例子。

首先是两个帮助函数,避免做一些重复劳动:
// 帮助函数,打印vector的内容
template <typename T>
std::ostream &operator<<(std::ostream &os, const std::vector<T> &v)
{
    os << "{ ";
    for (const auto &i: v) {
        os << i << ", ";
    }
    os << "}";

    return os;
}

// 帮助函数,生成一个符合UniformRandomBitGenerator要求的随机数生成器;
// std::random_device虽然符合要求但是只适合于生成seed及安全要求较高的场合,因为速度可能很慢。
// 所以我们选择std::mt19937算法,你可以自己选择其他合适的算法
auto get_URBG()
{
    std::random_device rd;
    // 使用random_device生成seed
    std::mt19937 g(rd());

    return g;
}

然后我们分别打乱一个std::vector<int>和std::vector<std::string>容器内元素的排列顺序:
// 随机排序容器内元素,打印随机排序前和随机排序后的容器内容
template <typename T>
void shuffle_container(std::vector<T> &container)
{
    std::cout << "before shuffle: " << container << std::endl;
    std::shuffle(container.begin(), container.end(), get_URBG());
    std::cout << "after shuffle: " << container << std::endl;
}

int main()
{
    std::vector<int> ivec{1,2,3,4,5};
    shuffle_container(ivec);

    // 分割线
    std::cout << std::string(40, '-') << std::endl;

    std::vector<std::string> svec{"a", "b", "c", "d", "e", "f", "g"};
    shuffle_container(svec);
}

我们编译写好的程序,然后运行:

可以看到元素都已经被随机排序了。

linux
相关资讯       C++ 
本文评论   查看全部评论 (0)
表情: 表情 姓名: 字数

       

评论声明
  • 尊重网上道德,遵守中华人民共和国的各项有关法律法规
  • 承担一切因您的行为而直接或间接导致的民事或刑事法律责任
  • 本站管理人员有权保留或删除其管辖留言中的任意内容
  • 本站有权在网站内转载或引用您的评论
  • 参与本评论即表明您已经阅读并接受上述条款