ITK 6.0.0
Insight Toolkit
 
Loading...
Searching...
No Matches
itkMersenneTwisterRandomVariateGenerator.h
Go to the documentation of this file.
1/*=========================================================================
2 *
3 * Copyright NumFOCUS
4 *
5 * Licensed under the Apache License, Version 2.0 (the "License");
6 * you may not use this file except in compliance with the License.
7 * You may obtain a copy of the License at
8 *
9 * https://www.apache.org/licenses/LICENSE-2.0.txt
10 *
11 * Unless required by applicable law or agreed to in writing, software
12 * distributed under the License is distributed on an "AS IS" BASIS,
13 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14 * See the License for the specific language governing permissions and
15 * limitations under the License.
16 *
17 *=========================================================================*/
18#ifndef itkMersenneTwisterRandomVariateGenerator_h
19#define itkMersenneTwisterRandomVariateGenerator_h
20
21#include "itkMacro.h"
22#include "itkObjectFactory.h"
24#include "itkIntTypes.h"
25#include "itkMath.h"
26#include "itkSingletonMacro.h"
27
28#include <atomic>
29#include <mutex>
30#include <climits>
31#include <ctime>
32#include <limits>
33
34namespace itk
35{
36namespace Statistics
37{
122
123struct MersenneTwisterGlobals;
124
126{
127public:
133
134 using IntegerType = uint32_t;
135
137 itkOverrideGetNameOfClassMacro(MersenneTwisterRandomVariateGenerator);
138
147 static Pointer
149
158 static Pointer
160
163 static void
165
167 static constexpr IntegerType StateVectorLength = 624;
168
170 void
171 Initialize(const IntegerType seed);
172
173 /* Initialize with clock time */
174 void
175 Initialize();
176
178 double
180
182 double
183 GetVariateWithClosedRange(const double n);
184
186 double
188
190 double
191 GetVariateWithOpenUpperRange(const double n);
192
194 double
196
198 double
199 GetVariateWithOpenRange(const double n);
200
204
208
211 double
213
214 /* Access to a normal random number distribution
215 * TODO: Compare with vnl_sample_normal */
216 double
217 GetNormalVariate(const double mean = 0.0, const double variance = 1.0);
218
219 /* Access to a uniform random number distribution in the range [a, b)
220 * TODO: Compare with vnl_sample_uniform */
221 double
222 GetUniformVariate(const double a, const double b);
223
229 double
230 GetVariate() override;
231
233 double
234 operator()();
235
241 inline void
242 SetSeed(const IntegerType oneSeed);
243 inline void
244 SetSeed();
251 GetSeed() const;
252
258 static IntegerType
260
261 /*
262 // Saving and loading generator state
263 void save( IntegerType* saveArray ) const; // to array of size SAVE
264 void load( IntegerType *const loadArray ); // from such array
265 */
266
267protected:
270 void
271 PrintSelf(std::ostream & os, Indent indent) const override;
272
274 void
275 reload();
276
278 hiBit(const IntegerType & u) const
279 {
280 return u & 0x80000000;
281 }
282 IntegerType
283 loBit(const IntegerType & u) const
284 {
285 return u & 0x00000001;
286 }
287 IntegerType
288 loBits(const IntegerType & u) const
289 {
290 return u & 0x7fffffff;
291 }
292 IntegerType
293 mixBits(const IntegerType & u, const IntegerType & v) const
294 {
295 return hiBit(u) | loBits(v);
296 }
297
298 IntegerType
299 twist(const IntegerType & m, const IntegerType & s0, const IntegerType & s1) const
300 {
301 return m ^ (mixBits(s0, s1) >> 1) ^ (-static_cast<int32_t>(loBit(s1)) & 0x9908b0df);
302 }
303
304 static IntegerType
305 hash(time_t t, clock_t c);
306
307private:
309 itkGetGlobalDeclarationMacro(MersenneTwisterGlobals, PimplGlobals);
310
312 static Pointer
314
315 // Internal state
317
318 // Next value to get from state
320
321 // Number of values left before reload is needed
322 int m_Left{};
323
324 // Seed value
325 std::atomic<IntegerType> m_Seed{};
326
327 // Local lock to enable concurrent access to singleton
328 std::mutex m_InstanceMutex{};
329
330 // Static/Global Variable need to be thread-safely accessed
331
332 static MersenneTwisterGlobals * m_PimplGlobals;
333
334}; // end of class
335
336// Declare inlined functions.... (must be declared in the header)
337
338inline void
340{
341 const std::lock_guard<std::mutex> lockGuard(m_InstanceMutex);
342 this->m_Seed = seed;
343 // Initialize generator state with seed
344 // See Knuth TAOCP Vol 2, 3rd Ed, p.106 for multiplier.
345 // In previous versions, most significant bits (MSBs) of the seed affect
346 // only MSBs of the state array. Modified 9 Jan 2002 by Makoto Matsumoto.
347 IntegerType * s = state;
348 IntegerType * r = state;
349
350 *s++ = seed;
352 {
353 *s++ = uint32_t{ (uint32_t{ 1812433253 } * (*r ^ (*r >> 30)) + i) };
354 ++r;
355 }
356 reload();
357}
358
359inline void
361{
362 // Generate N new values in state
363 // Made clearer and faster by Matthew Bellew
364 // matthew dot bellew at home dot com
365
366 // Period parameter
367 static constexpr unsigned int M = 397;
368
369 // get rid of VS warning
370 constexpr auto index = int{ M } - int{ MersenneTwisterRandomVariateGenerator::StateVectorLength };
371
372 IntegerType * p = state;
373
375 {
376 *p = twist(p[M], p[0], p[1]);
377 }
378 for (int i = M; --i; ++p)
379 {
380 *p = twist(p[index], p[0], p[1]);
381 }
382 *p = twist(p[index], p[0], state[0]);
383
385 m_PNext = state;
386}
387
388inline void
393
394inline void
396{
397 // Seed the generator with a simple IntegerType
398 Initialize(oneSeed);
399}
400
401inline void
403{
404 // use time() and clock() to generate a unlikely-to-repeat seed.
405 SetSeed(hash(time(nullptr), clock()));
406}
407
408
411{
412 return this->m_Seed;
413}
414
418{
419 if (m_Left == 0)
420 {
421 reload();
422 }
423 --m_Left;
424
425 IntegerType s1 = *m_PNext++;
426 s1 ^= (s1 >> 11);
427 s1 ^= (s1 << 7) & 0x9d2c5680;
428 s1 ^= (s1 << 15) & 0xefc60000;
429 return (s1 ^ (s1 >> 18));
430}
431
432inline double
434{
435 return static_cast<double>(GetIntegerVariate()) * (1.0 / double{ std::numeric_limits<uint32_t>::max() });
436}
437
439inline double
444
446inline double
448{
449 return static_cast<double>(GetIntegerVariate()) / double{ 1ULL << 32ULL };
450}
451
453inline double
458
460inline double
462{
463 return (static_cast<double>(GetIntegerVariate()) + 0.5) / double{ 1ULL << 32ULL };
464}
465
467inline double
472
475{
476 // Find which bits are used in n
477 IntegerType used = n;
478
479 used |= used >> 1;
480 used |= used >> 2;
481 used |= used >> 4;
482 used |= used >> 8;
483 used |= used >> 16;
484
485 // Draw numbers until one is found in [0,n]
486 IntegerType i;
487 do
488 {
489 i = GetIntegerVariate() & used; // toss unused bits to shorten search
490 } while (i > n);
491
492 return i;
493}
494
497inline double
499{
500 const IntegerType a = GetIntegerVariate() >> 5;
501 const IntegerType b = GetIntegerVariate() >> 6;
502
503 return (a * 67108864.0 + b) * (1.0 / 9007199254740992.0); // by Isaku
504 // Wada
505}
506
508// TODO: Compare with vnl_sample_normal
509inline double
510MersenneTwisterRandomVariateGenerator::GetNormalVariate(const double mean, const double variance)
511{
512 // Return a real number from a normal (Gaussian) distribution with given
513 // mean and variance by Box-Muller method
514 const double r = std::sqrt(-2.0 * std::log(1.0 - GetVariateWithOpenRange()) * variance);
515 const double phi = 2.0 * itk::Math::pi * GetVariateWithOpenUpperRange();
516
517 return mean + r * std::cos(phi);
518}
519
521// TODO: Compare with vnl_sample_uniform
522inline double
524{
525 const double u = GetVariateWithOpenUpperRange();
526
527 return ((1.0 - u) * a + u * b);
528}
529
530inline double
535
536inline double
541
542/* Change log from MTRand.h */
543// Change log:
544//
545// v0.1 - First release on 15 May 2000
546// - Based on code by Makoto Matsumoto, Takuji Nishimura, and Shawn Cokus
547// - Translated from C to C++
548// - Made completely ANSI compliant
549// - Designed convenient interface for initialization, seeding, and
550// obtaining numbers in default or user-defined ranges
551// - Added automatic seeding from /dev/urandom or time() and clock()
552// - Provided functions for saving and loading generator state
553//
554// v0.2 - Fixed bug which reloaded generator one step too late
555//
556// v0.3 - Switched to clearer, faster reload() code from Matthew Bellew
557//
558// v0.4 - Removed trailing newline in saved generator format to be consistent
559// with output format of built-in types
560//
561// v0.5 - Improved portability by replacing static const int's with enum's and
562// clarifying return values in seed(); suggested by Eric Heimburg
563// - Removed MAXINT constant; use 0xffffffffUL instead
564//
565// v0.6 - Eliminated seed overflow when uint32 is larger than 32 bits
566// - Changed integer [0,n] generator to give better uniformity
567//
568// v0.7 - Fixed operator precedence ambiguity in reload()
569// - Added access for real numbers in (0,1) and (0,n)
570//
571// v0.8 - Included time.h header to properly support time_t and clock_t
572//
573// v1.0 - Revised seeding to match 26 Jan 2002 update of Nishimura and Matsumoto
574// - Allowed for seeding with arrays of any length
575// - Added access for real numbers in [0,1) with 53-bit resolution
576// - Added access for real numbers from normal (Gaussian) distributions
577// - Increased overall speed by optimizing twist()
578// - Doubled speed of integer [0,n] generation
579// - Fixed out-of-range number generation on 64-bit machines
580// - Improved portability by substituting literal constants for long enum's
581// - Changed license from GNU LGPL to BSD
582} // end namespace Statistics
583} // end namespace itk
584
585#endif
Control indentation during Print() invocation.
Definition itkIndent.h:50
Implements transparent reference counting.
IntegerType mixBits(const IntegerType &u, const IntegerType &v) const
static Pointer New()
Method for creation through the object factory.
void PrintSelf(std::ostream &os, Indent indent) const override
IntegerType twist(const IntegerType &m, const IntegerType &s0, const IntegerType &s1) const
static IntegerType hash(time_t t, clock_t c)
itkGetGlobalDeclarationMacro(MersenneTwisterGlobals, PimplGlobals)
double GetNormalVariate(const double mean=0.0, const double variance=1.0)
static constexpr double pi
Definition itkMath.h:66
The "itk" namespace contains all Insight Segmentation and Registration Toolkit (ITK) classes....