source: src/RandomNumbers/RandomNumberGenerator_Encapsulation.hpp@ 485eea

ForceAnnealing_goodresults ForceAnnealing_tocheck
Last change on this file since 485eea was 63839f, checked in by Frederik Heber <heber@…>, 15 years ago

RandomNumberGeneratorFactory does not clone anymore.

  • cloning of the generator was nonsense, as (because of same seed), we always start at same random number sequence. I.e. if we often obtain new random number generator we hardly get random numbers.
  • RandomNumberGeneratorFactory::makeRandomNumberGenerator() returns now reference to avoid accidental free'ing of the instance (that is still contained in the prototype table of the factory).
  • unit test of RandomNumberGenerator now checks for non-copy received.
  • RandomNumberGenerator()::get...Name() are now const members.
  • in many cases where pointer *rng was received and ref &random obtained from it, we now receive ref directly (and then don't accidentally forget to delete the clone anymore. Which is good in any case!).
  • Property mode set to 100644
File size: 6.5 KB
Line 
1/*
2 * RandomNumberGenerator_Encapsulation.hpp
3 *
4 * Created on: Dec 31, 2010
5 * Author: heber
6 */
7
8#ifndef RANDOMNUMBERGENERATOR_ENCAPSULATION_HPP_
9#define RANDOMNUMBERGENERATOR_ENCAPSULATION_HPP_
10
11// include config.h
12#ifdef HAVE_CONFIG_H
13#include <config.h>
14#endif
15
16#include <boost/nondet_random.hpp>
17#include <boost/random.hpp>
18#include <boost/random/bernoulli_distribution.hpp>
19#include <boost/random/binomial_distribution.hpp>
20#include <boost/random/cauchy_distribution.hpp>
21#include <boost/random/exponential_distribution.hpp>
22#include <boost/random/gamma_distribution.hpp>
23#include <boost/random/geometric_distribution.hpp>
24#include <boost/random/linear_congruential.hpp>
25#include <boost/random/lognormal_distribution.hpp>
26#include <boost/random/normal_distribution.hpp>
27#include <boost/random/poisson_distribution.hpp>
28#include <boost/random/triangle_distribution.hpp>
29#include <boost/random/uniform_01.hpp>
30#include <boost/random/uniform_int.hpp>
31#include <boost/random/uniform_on_sphere.hpp>
32#include <boost/random/uniform_real.hpp>
33#include <boost/random/uniform_smallint.hpp>
34#include <boost/random/additive_combine.hpp>
35#include <boost/random/discard_block.hpp>
36#include <boost/random/inversive_congruential.hpp>
37#include <boost/random/lagged_fibonacci.hpp>
38#include <boost/random/linear_congruential.hpp>
39#include <boost/random/linear_feedback_shift.hpp>
40#include <boost/random/mersenne_twister.hpp>
41#include <boost/random/random_number_generator.hpp>
42#include <boost/random/ranlux.hpp>
43#include <boost/random/shuffle_output.hpp>
44#include <boost/random/subtract_with_carry.hpp>
45#include <boost/random/xor_combine.hpp>
46#include <boost/random/variate_generator.hpp>
47
48#include <typeinfo>
49
50#include "CodePatterns/Clone.hpp"
51#include "RandomNumberDistribution_Encapsulation.hpp"
52#include "RandomNumberDistributionFactory.hpp"
53#include "RandomNumberEngine_Encapsulation.hpp"
54#include "RandomNumberEngineFactory.hpp"
55#include "RandomNumberGenerator.hpp"
56
57class RandomNumberGeneratorFactory;
58
59/** Template class that encapsulates the random number generators from
60 * random::boost.
61 *
62 * We inherit the interface RandomNumberGenerator such that all are
63 * accessible in the same way (i.e. RandomNumberGeneratorFactory will
64 * spit out only references to RandomNumberGenerator).
65 *
66 * Note that we always returns double values although the distribution
67 * might be integer or even a discrete distribution of integers.
68 *
69 * We need two template parameters:
70 * -# the engine - generates uniform random numbers
71 * -# the distribution - transforms uniform into a desired distribution
72 */
73template <class engine, class distribution>
74class RandomNumberGenerator_Encapsulation :
75 public RandomNumberGenerator,
76 public Clone<RandomNumberGenerator>
77{
78 /**
79 * Factory is friend such that it can access private cstor when filling its
80 * table
81 */
82 friend class RandomNumberGeneratorFactory;
83
84public:
85 /** obtain a random number via generator and the specific distribution.
86 *
87 */
88 double operator()() const {
89 const double value = (*randomgenerator)();
90 //std::cout << "Current random value from (" << EngineName() << "," << DistributionName() << ") is " << value << std::endl;
91 return value;
92 }
93
94 /** Set the generator's seed.
95 *
96 * @param _seed seed to set to
97 */
98 void seed(unsigned int _seed) {
99 engine_type->seed(_seed);
100 }
101
102 /** Getter for smallest possible random number
103 *
104 * @return smallest possible random number
105 */
106 double min() const {
107 return distribution_type->min();
108 }
109
110 /** Getter for largest possible random number
111 *
112 * @return largest possible random number
113 */
114 double max() const {
115 return distribution_type->max();
116 }
117
118 /** Getter for the type name of the internal engine.
119 *
120 */
121 std::string EngineName() const {
122 return engine_type->name();
123 }
124
125 /** Getter for the type name of the internal distribution.
126 *
127 */
128 std::string DistributionName() const {
129 return distribution_type->name();
130 }
131
132 /** Clones the current instance and returns pointer.
133 *
134 * @return pointer to cloned instance
135 */
136 RandomNumberGenerator* clone() const
137 {
138 RandomNumberGenerator_Encapsulation<engine, distribution> *MyClone = NULL;
139
140 // sadly (due to construction of variate_generator without any abstract
141 // base class) we need RTTI here ...
142 RandomNumberEngine *engine_clone =
143 RandomNumberEngineFactory::getInstance().getProduct(typeid(engine));
144 RandomNumberDistribution *distribution_clone =
145 RandomNumberDistributionFactory::getInstance().getProduct(typeid(distribution));
146
147 MyClone = new RandomNumberGenerator_Encapsulation<engine, distribution>(
148 engine_clone,
149 distribution_clone
150 );
151
152 return MyClone;
153 }
154
155protected:
156 /** Constructor that instantiates a specific random number generator and
157 * distribution.
158 *
159 * This is one is supposed to create the prototypes. Hence, must only be
160 * called from the factory.
161 */
162 RandomNumberGenerator_Encapsulation() :
163 randomgenerator(NULL)
164 {
165 engine_type = RandomNumberEngineFactory::getInstance().getProduct(typeid(engine));
166 distribution_type = RandomNumberDistributionFactory::getInstance().getProduct(typeid(distribution));
167 randomgenerator = new boost::variate_generator<engine, distribution>(
168 (dynamic_cast<RandomNumberEngine_Encapsulation<engine> *>(engine_type))
169 ->getEngine(),
170 (dynamic_cast<RandomNumberDistribution_Encapsulation<distribution> *>(distribution_type))
171 ->getDistribution());
172 }
173
174 /** Constructor that instantiates a specific random number generator and
175 * distribution.
176 * @param _engine_type instance of the desired generator
177 * @param _distribution_type instance of the desired distribution
178 */
179 RandomNumberGenerator_Encapsulation(
180 RandomNumberEngine *_engine_type,
181 RandomNumberDistribution *_distribution_type
182 ) :
183 randomgenerator(NULL)
184 {
185 engine_type = _engine_type;
186 distribution_type = _distribution_type;
187 randomgenerator = new boost::variate_generator<engine, distribution> (
188 (dynamic_cast<RandomNumberEngine_Encapsulation<engine> *>(engine_type))
189 ->getEngine(),
190 (dynamic_cast<RandomNumberDistribution_Encapsulation<distribution> *>(distribution_type))
191 ->getDistribution());
192 }
193
194 /** Destructor of the class.
195 *
196 */
197 virtual ~RandomNumberGenerator_Encapsulation()
198 {
199 // NULL pointer may be deleted
200 delete randomgenerator;
201 }
202private:
203 mutable boost::variate_generator<engine, distribution> *randomgenerator;
204};
205
206#endif /* RANDOMNUMBERGENERATOR_ENCAPSULATION_HPP_ */
Note: See TracBrowser for help on using the repository browser.