All Downloads are FREE. Search and download functionalities are using the official Maven repository.

spire.random.Ziggurat.scala Maven / Gradle / Ivy

/************************************************************************\
** Project                                                              **
**       ______  ______   __    ______    ____                          **
**      / ____/ / __  /  / /   / __  /   / __/     (c) 2011-2014        **
**     / /__   / /_/ /  / /   / /_/ /   / /_                            **
**    /___  / / ____/  / /   / __  /   / __/   Erik Osheim, Tom Switzer **
**   ____/ / / /      / /   / / | |   / /__                             **
**  /_____/ /_/      /_/   /_/  |_|  /____/     All rights reserved.    **
**                                                                      **
**      Redistribution and use permitted under the MIT license.         **
**                                                                      **
\************************************************************************/


package spire
package random

import scala.annotation.tailrec

/**
 * This is a Scala implementation of the Ziggurat algorithm for generating random variables from decreasing densities.
 *
 * 

Reference: * George Marsaglia, Wai Wan Tsang: * "The Ziggurat Method for Generating Random Variables" * Journal of Statistical Software, Vol. 5, Issue 8, October 2000. * * @see Ziggurat Paper * @see Ziggurat algorithm @ Wikipedia * @author Dušan Kysel */ object Ziggurat { import scala.math.exp import scala.math.sqrt import scala.math.log import scala.math.abs private val kn = new Array[Long](128) private val wn = new Array[Double](128) private val fn = new Array[Double](128) private val ke = new Array[Long](256) private val we = new Array[Double](256) private val fe = new Array[Double](256) def rnor(g: Generator): Double = { val hz = g.nextInt() val iz = hz & 127 if (abs(hz) < kn(iz)) hz * wn(iz) else nfix(g, hz, iz) } def rexp(g: Generator): Double = { val jz = g.nextInt() & 0xffffffffL val iz = (jz & 255).toInt if (jz < ke(iz)) jz * we(iz) else efix(g, jz, iz) } private def nfix(g: Generator, hza: Int, iza: Int) : Double = { val r = 3.442619855899d val r1 = 1 / r var x, y = 0d var hz = hza var iz = iza @tailrec def loop(): Double = { x = hz * wn(iz) if (iz == 0) { do { x = -log(g.nextDouble()) * r1 y = -log(g.nextDouble()) } while (y + y < x * x) return if (hz > 0) r + x else -r - x } if (fn(iz) + g.nextDouble() * (fn(iz - 1) - fn(iz)) < exp(-.5 * x * x)) return x hz = g.nextInt() iz = hz & 127 if (abs(hz) < kn(iz)) return hz * wn(iz) loop() } loop() } private def efix(g: Generator, jza: Long, iza: Int) : Double = { var jz = jza var iz = iza @tailrec def loop(): Double = { if (iz == 0) return 7.697117470131487 - log(g.nextDouble()) val x = jz * we(iz) if (fe(iz) + g.nextDouble() * (fe(iz - 1) - fe(iz)) < exp(-x)) return x jz = g.nextInt() & 0xffffffffL iz = (jz & 255).toInt if (jz < ke(iz)) return jz * we(iz) loop() } loop() } { val m1 : Double = 2147483648d val m2 : Double = 4294967296d var dn : Double = 3.442619855899 var tn : Double = dn var de : Double = 7.697117470131487 var te : Double = de val vn : Double = 9.91256303526217e-3 val ve : Double = 3.949659822581572e-3 var q : Double = vn / exp(-.5 * dn * dn) kn(0) = ((dn / q) * m1).toLong kn(1) = 0 wn(0) = q / m1 wn(127) = dn / m1 fn(0) = 1d fn(127) = exp(-.5 * dn * dn) for (i <- 126 to 1 by -1) { dn = sqrt(-2 * log(vn / dn + exp(-.5 * dn * dn))) kn(i+1) = ((dn / tn) * m1).toLong tn = dn fn (i) = exp(-.5 * dn * dn) wn(i) = dn / m1 } q = ve / exp(-de) ke(0) = ((de / q) * m2).toLong ke(1) = 0 we(0) = q / m2 we(255) = de / m2 fe(0) = 1d fe(255) = exp(-de) for (i <- 254 to 1 by -1) { de = -log(ve / de + exp(-de)) ke(i+1) = ((de / te) * m2).toLong te = de fe (i) = exp(-de) we(i) = de / m2 } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy