#region License Information
/* HeuristicLab
* Copyright (C) Heuristic and Evolutionary Algorithms Laboratory (HEAL)
*
* This file is part of HeuristicLab.
*
* HeuristicLab 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 3 of the License, or
* (at your option) any later version.
*
* HeuristicLab 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 for more details.
*
* You should have received a copy of the GNU General Public License
* along with HeuristicLab. If not, see .
*/
#endregion
using System;
using HeuristicLab.Common;
using HeuristicLab.Core;
using HEAL.Attic;
namespace HeuristicLab.Random {
///
/// Normally distributed random variable.
/// Uses Marsaglia's polar method
///
[Item("NormalDistributedRandomPolar", "A pseudo random number generator which uses Marsaglia's polar method to create normally distributed random numbers.")]
[StorableType("B17E35DB-1DC6-434A-8F08-4AD4AB224C59")]
public sealed class NormalDistributedRandomPolar : Item, IRandom {
[Storable]
private double mu;
///
/// Gets or sets the value for µ.
///
public double Mu {
get { return mu; }
set { mu = value; }
}
[Storable]
private double sigma;
///
/// Gets or sets the value for sigma.
///
public double Sigma {
get { return sigma; }
set { sigma = value; }
}
[Storable]
private IRandom uniform;
[StorableConstructor]
private NormalDistributedRandomPolar(StorableConstructorFlag _) : base(_) { }
private NormalDistributedRandomPolar(NormalDistributedRandomPolar original, Cloner cloner)
: base(original, cloner) {
uniform = cloner.Clone(original.uniform);
mu = original.mu;
sigma = original.sigma;
}
///
/// Initializes a new instance of with µ = 0 and sigma = 1
/// and a new random number generator.
///
public NormalDistributedRandomPolar() {
this.mu = 0.0;
this.sigma = 1.0;
this.uniform = new MersenneTwister();
}
///
/// Initializes a new instance of with the given parameters.
/// The random number generator is not copied!
///
/// The random number generator.
/// The value for µ.
/// The value for sigma.
public NormalDistributedRandomPolar(IRandom uniformRandom, double mu, double sigma) {
this.mu = mu;
this.sigma = sigma;
this.uniform = uniformRandom;
}
#region IRandom Members
///
public void Reset() {
uniform.Reset();
}
///
public void Reset(int seed) {
uniform.Reset(seed);
}
///
/// This method is not implemented.
///
public int Next() {
throw new NotImplementedException();
}
///
/// This method is not implemented.
///
public int Next(int maxVal) {
throw new NotImplementedException();
}
///
/// This method is not implemented.
///
public int Next(int minVal, int maxVal) {
throw new NotImplementedException();
}
///
/// Generates a new double random number.
///
/// A double random number.
public double NextDouble() {
return NormalDistributedRandomPolar.NextDouble(uniform, mu, sigma);
}
#endregion
///
/// Clones the current instance (deep clone).
///
/// The cloned object as .
public override IDeepCloneable Clone(Cloner cloner) {
return new NormalDistributedRandomPolar(this, cloner);
}
/**
* Polar method due to Marsaglia.
*
* Devroye, L. Non-Uniform Random Variates Generation. Springer-Verlag,
* New York, 1986, Ch. V, Sect. 4.4.
*/
public static double NextDouble(IRandom uniformRandom, double mu, double sigma) {
// we don't use spare numbers (efficency loss but easier for multi-threaded code)
double u, v, s;
do {
u = uniformRandom.NextDouble() * 2 - 1;
v = uniformRandom.NextDouble() * 2 - 1;
s = u * u + v * v;
} while (s > 1 || s == 0);
s = Math.Sqrt(-2.0 * Math.Log(s) / s);
return mu + sigma * u * s;
}
}
}