78 lines
3.3 KiB
C++
78 lines
3.3 KiB
C++
/****************************************************************************
|
|
* VCGLib o o *
|
|
* Visual and Computer Graphics Library o o *
|
|
* _ O _ *
|
|
* Copyright(C) 2004 \/)\/ *
|
|
* Visual Computing Lab /\/| *
|
|
* ISTI - Italian National Research Council | *
|
|
* \ *
|
|
* All rights reserved. *
|
|
* *
|
|
* This program is free software; you can redistribute it and/or modify *
|
|
* it under the terms of the GNU General Public License as published by *
|
|
* the Free Software Foundation; either version 2 of the License, or *
|
|
* (at your option) any later version. *
|
|
* *
|
|
* This program is distributed in the hope that it will be useful, *
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of *
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
|
|
* GNU General Public License (http://www.gnu.org/licenses/gpl.txt) *
|
|
* for more details. *
|
|
* *
|
|
****************************************************************************/
|
|
|
|
// RandomGenerator is derived from a STL extension of sgi:
|
|
// it is based on the Subtractive Ring method.
|
|
// See section 3.6 of Knuth for an implementation of the subtractive method in FORTRAN.
|
|
// Section 3.2.2 of Knuth analyzes this class of algorithms.
|
|
// (D. E. Knuth, The Art of Computer Programming. Volume 2: Seminumerical Algorithms, second edition. Addison-Wesley, 1981.) .
|
|
// Note: this code assumes that int is 32 bits.
|
|
|
|
#ifndef __VCG_RandomGenerator
|
|
#define __VCG_RandomGenerator
|
|
|
|
#include <functional>
|
|
namespace vcg {
|
|
namespace math {
|
|
|
|
class RandomGenerator : public std::unary_function<unsigned int, unsigned int> {
|
|
private:
|
|
unsigned int _M_table[55];
|
|
size_t _M_index1;
|
|
size_t _M_index2;
|
|
public:
|
|
unsigned int operator()(unsigned int __limit) {
|
|
_M_index1 = (_M_index1 + 1) % 55;
|
|
_M_index2 = (_M_index2 + 1) % 55;
|
|
_M_table[_M_index1] = _M_table[_M_index1] - _M_table[_M_index2];
|
|
return _M_table[_M_index1] % __limit;
|
|
}
|
|
|
|
void _M_initialize(unsigned int __seed)
|
|
{
|
|
unsigned int __k = 1;
|
|
_M_table[54] = __seed;
|
|
size_t __i;
|
|
for (__i = 0; __i < 54; __i++) {
|
|
size_t __ii = (21 * (__i + 1) % 55) - 1;
|
|
_M_table[__ii] = __k;
|
|
__k = __seed - __k;
|
|
__seed = _M_table[__ii];
|
|
}
|
|
for (int __loop = 0; __loop < 4; __loop++) {
|
|
for (__i = 0; __i < 55; __i++)
|
|
_M_table[__i] = _M_table[__i] - _M_table[(1 + __i + 30) % 55];
|
|
}
|
|
_M_index1 = 0;
|
|
_M_index2 = 31;
|
|
}
|
|
|
|
RandomGenerator(unsigned int __seed) { _M_initialize(__seed); }
|
|
RandomGenerator() { _M_initialize(161803398u); }
|
|
|
|
};
|
|
} // end namespace math
|
|
} // end namespace vcg
|
|
|
|
|
|
#endif |