forked from andrewkirillov/AForge.NET
-
Notifications
You must be signed in to change notification settings - Fork 22
/
Copy pathStandardGenerator.cs
132 lines (117 loc) · 3.54 KB
/
StandardGenerator.cs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
// AForge Math Library
// AForge.NET framework
// http://www.aforgenet.com/framework/
//
// Copyright © AForge.NET, 2007-2011
// contacts@aforgenet.com
//
namespace AForge.Math.Random
{
using System;
/// <summary>
/// Standard random numbers generator.
/// </summary>
///
/// <remarks><para>The random number generator generates gaussian
/// random numbers with zero mean and standard deviation of one. The generator
/// implements polar form of the Box-Muller transformation.</para>
///
/// <para>The generator uses <see cref="UniformOneGenerator"/> generator as a base
/// to generate random numbers.</para>
///
/// <para>Sample usage:</para>
/// <code>
/// // create instance of random generator
/// IRandomNumberGenerator generator = new StandardGenerator( );
/// // generate random number
/// float randomNumber = generator.Next( );
/// </code>
/// </remarks>
///
public class StandardGenerator : IRandomNumberGenerator
{
private UniformOneGenerator rand = null;
private float secondValue;
private bool useSecond = false;
/// <summary>
/// Mean value of the generator.
/// </summary>
///
public float Mean
{
get { return 0; }
}
/// <summary>
/// Variance value of the generator.
/// </summary>
///
public float Variance
{
get { return 1; }
}
/// <summary>
/// Initializes a new instance of the <see cref="StandardGenerator"/> class.
/// </summary>
///
public StandardGenerator( )
{
rand = new UniformOneGenerator( );
}
/// <summary>
/// Initializes a new instance of the <see cref="StandardGenerator"/> class.
/// </summary>
///
/// <param name="seed">Seed value to initialize random numbers generator.</param>
///
public StandardGenerator( int seed )
{
rand = new UniformOneGenerator( seed );
}
/// <summary>
/// Generate next random number.
/// </summary>
///
/// <returns>Returns next random number.</returns>
///
public float Next( )
{
// check if we can use second value
if ( useSecond )
{
// return the second number
useSecond = false;
return secondValue;
}
float x1, x2, w, firstValue;
// generate new numbers
do
{
x1 = (float) rand.Next( ) * 2.0f - 1.0f;
x2 = (float) rand.Next( ) * 2.0f - 1.0f;
w = x1 * x1 + x2 * x2;
}
while ( w >= 1.0f );
w = (float) Math.Sqrt( ( -2.0f * Math.Log( w ) ) / w );
// get two standard random numbers
firstValue = x1 * w;
secondValue = x2 * w;
useSecond = true;
// return the first number
return firstValue;
}
/// <summary>
/// Set seed of the random numbers generator.
/// </summary>
///
/// <param name="seed">Seed value.</param>
///
/// <remarks>Resets random numbers generator initializing it with
/// specified seed value.</remarks>
///
public void SetSeed( int seed )
{
rand = new UniformOneGenerator( seed );
useSecond = false;
}
}
}