JZo*_*ker 1 c++ random generator
如何编写代码以生成介于两个不同范围之间的随机数?
示例:生成一个介于5到7之间或介于10到12之间的随机数.可能的结果是5,6,7,10,11或12.
seh*_*ehe 10
更新下面添加的第二个实现
如果你想使这个通用,你将不得不编码.
我想到两个选择:
int arr[]={5,6,7,10,11,12}第二:
#include <random>
#include <iostream>
int main()
{
using namespace std;
vector<int> arr = {5,6,7,10,11,12};
mt19937 prng { random_device {} () };
uniform_int_distribution<> dist(0, arr.size()-1);
int i = 10;
while (i--)
std::cout << arr[dist(prng)] << " ";
}
Run Code Online (Sandbox Code Playgroud)
打印
5 5 6 12 11 6 11 12 5 12
Run Code Online (Sandbox Code Playgroud)
或者,当然类似
通过使用Boost Interval Container Library有效地表示构成域的间隔,可以扩展许多段或大段的替代实现:
template <typename T = int>
struct uniform_draw {
using set = boost::icl::interval_set<T>;
using ival = typename set::interval_type::type;
uniform_draw(std::initializer_list<ival> data)
: set_(make_set(data)), dist_(0, set_.size() - 1)
{ }
friend std::ostream& operator<<(std::ostream& os, uniform_draw const& ud) {
return os << ud.set_ << " (#samples:" << ud.set_.size() << ")";
}
template <typename Engine>
T operator()(Engine& engine) {
uintmax_t index = dist_(engine);
std::cout << " - index: " << index << " against " << set_ << "\n";
// I think this can be optimized. I just don't know how to elegantly do that / yet
for (auto& iv : set_) {
std::cout << " - index: " << index << " against " << iv << "\n";
if (index > size(iv)) {
index -= size(iv);
} else {
return iv.lower() + index;
}
}
throw std::range_error("uniform_draw");
}
private:
set make_set(std::initializer_list<ival> data) {
set r;
for (auto& el : data)
r.insert(el);
return r;
}
set const set_;
std::uniform_int_distribution<T> dist_; // TODO make_unsigned<T>?
};
Run Code Online (Sandbox Code Playgroud)
像往常一样使用它:
mt19937 mt { random_device {} () };
uniform_draw<int> dist { {5, 7}, {10, 12} };
std::cout << dist << "\n";
for (int i = 0; i < 10; ++i)
std::cout << "RESULT: " << dist(mt) << "\n";
Run Code Online (Sandbox Code Playgroud)
打印例如:
{[5,7)[10,12)} (#samples:4)
7 7 6 11 6 6 7 7 7 6
Run Code Online (Sandbox Code Playgroud)