lin
randoms.hpp
Go to the documentation of this file.
1 // vim: set tabstop=2:softtabstop=2:shiftwidth=2:expandtab
2 
8 #ifndef LIN_GENERATORS_RANDOMS_HPP_
9 #define LIN_GENERATORS_RANDOMS_HPP_
10 
11 #include "../core.hpp"
12 
13 #include <cstdint>
14 #include <type_traits>
15 
16 namespace lin {
17 namespace internal {
18 
27  private:
30  unsigned long long seed;
31 
36  double cached_rand;
37 
42  bool has_cached;
43 
44  public:
45  constexpr RandomsGenerator(RandomsGenerator const &) = default;
46  constexpr RandomsGenerator(RandomsGenerator &&) = default;
47  constexpr RandomsGenerator &operator=(RandomsGenerator const &) = default;
48  constexpr RandomsGenerator &operator=(RandomsGenerator &&) = default;
49 
54  constexpr RandomsGenerator(unsigned long long seed = 0) :
55  seed(seed ^ 4101842887655102017LL),
56  cached_rand(0.0),
57  has_cached(false) { }
58 
63  constexpr double rand() {
64  seed ^= (seed >> 21);
65  seed ^= (seed << 35);
66  seed ^= (seed >> 4);
67 
68  return 5.42101086242752217E-20 * (seed * 2685821657736338717ULL);
69  }
70 
77  constexpr double gaussian() {
78  if(has_cached) {
79  has_cached = false;
80  return cached_rand;
81  }
82  else{
83  double R = std::sqrt(-2.0*std::log(rand()));
84  double T = 2.0*M_PI*rand();
85  cached_rand = R*std::sin(T);
86  has_cached = true;
87  return R*std::cos(T);
88  }
89  }
90 };
91 } // namespace internal
92 
107 template <class C, std::enable_if_t<internal::has_traits<C>::value, size_t> = 0>
108 constexpr auto rands(internal::RandomsGenerator &rand, size_t r = C::Traits::max_rows, size_t c = C::Traits::max_cols) {
109  typename C::Traits::eval_t t(r, c);
110  for (lin::size_t i = 0; i < t.size(); i++) t(i) = typename C::Traits::elem_t(rand.rand());
111  return t;
112 }
113 
129 template <class C, std::enable_if_t<internal::has_traits<C>::value, size_t> = 0>
130 constexpr auto gaussians(internal::RandomsGenerator &rand, size_t r = C::Traits::max_rows, size_t c = C::Traits::max_cols) {
131  typename C::Traits::eval_t t(r, c);
132  for (lin::size_t i = 0; i < t.size(); i++) t(i) = typename C::Traits::elem_t(rand.gaussian());
133  return t;
134 }
135 } // namespace lin
136 
137 #endif
bool has_cached
has_cached is true if there&#39;s a cached standard normal available.
Definition: randoms.hpp:42
constexpr double gaussian()
Return a gaussian random number with mean 0 and std 1. Uses Box-Mueller transform with caching...
Definition: randoms.hpp:77
constexpr double rand()
Generates a uniform random number in the range zero to one.
Definition: randoms.hpp:63
double cached_rand
Value of a cached standard normal gassauisan random.
Definition: randoms.hpp:36
constexpr RandomsGenerator(unsigned long long seed=0)
Creates a random number generator based on the specified seed.
Definition: randoms.hpp:54
std::size_t size_t
Type tracking tensor dimensions and sizing.
Definition: config.hpp:33
constexpr auto rands(internal::RandomsGenerator &rand, size_t r=C::Traits::max_rows, size_t c=C::Traits::max_cols)
Generates a Matrix or Vector populated with uniform random values between 0 and 1.
Definition: randoms.hpp:108
Random number generator.
Definition: randoms.hpp:26
unsigned long long seed
Seed for the random number generator.
Definition: randoms.hpp:30
Definition: config.hpp:27
constexpr auto gaussians(internal::RandomsGenerator &rand, size_t r=C::Traits::max_rows, size_t c=C::Traits::max_cols)
Generates a Matrix or Vector populated with independent gaussian random variables with a mean of 0 an...
Definition: randoms.hpp:130