Skip to content

Commit

Permalink
Merge pull request #5611 from fhalde/2.12.x
Browse files Browse the repository at this point in the history
Common usage of finding random number in a range [ci: last-only]
  • Loading branch information
lrytz authored Mar 28, 2017
2 parents 99492a2 + dd40147 commit f9a019c
Show file tree
Hide file tree
Showing 2 changed files with 138 additions and 0 deletions.
99 changes: 99 additions & 0 deletions src/library/scala/util/Random.scala
Original file line number Diff line number Diff line change
Expand Up @@ -43,11 +43,33 @@ class Random(val self: java.util.Random) extends AnyRef with Serializable {
*/
def nextDouble(): Double = self.nextDouble()

/** Returns the next pseudorandom, uniformly distributed double value
* between min (inclusive) and max (exclusive) from this random number generator's sequence.
*/
def between(minInclusive: Double, maxExclusive: Double): Double = {
require(minInclusive < maxExclusive, "Invalid bounds")

val next = nextDouble() * (maxExclusive - minInclusive) + minInclusive
if (next < maxExclusive) next
else Math.nextAfter(maxExclusive, Double.NegativeInfinity)
}

/** Returns the next pseudorandom, uniformly distributed float value
* between 0.0 and 1.0 from this random number generator's sequence.
*/
def nextFloat(): Float = self.nextFloat()

/** Returns the next pseudorandom, uniformly distributed float value
* between min (inclusive) and max (exclusive) from this random number generator's sequence.
*/
def between(minInclusive: Float, maxExclusive: Float): Float = {
require(minInclusive < maxExclusive, "Invalid bounds")

val next = nextFloat() * (maxExclusive - minInclusive) + minInclusive
if (next < maxExclusive) next
else Math.nextAfter(maxExclusive, Float.NegativeInfinity)
}

/** Returns the next pseudorandom, Gaussian ("normally") distributed
* double value with mean 0.0 and standard deviation 1.0 from this
* random number generator's sequence.
Expand All @@ -65,11 +87,88 @@ class Random(val self: java.util.Random) extends AnyRef with Serializable {
*/
def nextInt(n: Int): Int = self.nextInt(n)

/** Returns a pseudorandom, uniformly distributed int value between min
* (inclusive) and the specified value max (exclusive), drawn from this
* random number generator's sequence.
*/
def between(minInclusive: Int, maxExclusive: Int): Int = {
require(minInclusive < maxExclusive, "Invalid bounds")

val difference = maxExclusive - minInclusive
if (difference >= 0) {
nextInt(difference) + minInclusive
} else {
/* The interval size here is greater than Int.MaxValue,
* so the loop will exit with a probability of at least 1/2.
*/
def loop(): Int = {
val n = nextInt()
if (n >= minInclusive && n < maxExclusive) n
else loop()
}
loop()
}
}

/** Returns the next pseudorandom, uniformly distributed long value
* from this random number generator's sequence.
*/
def nextLong(): Long = self.nextLong()

/** Returns a pseudorandom, uniformly distributed long value between 0
* (inclusive) and the specified value (exclusive), drawn from this
* random number generator's sequence.
*/
def nextLong(n: Long): Long = {
require(n > 0, "n must be positive")

/*
* Divide n by two until small enough for nextInt. On each
* iteration (at most 31 of them but usually much less),
* randomly choose both whether to include high bit in result
* (offset) and whether to continue with the lower vs upper
* half (which makes a difference only if odd).
*/

var offset = 0L
var _n = n

while (_n >= Integer.MAX_VALUE) {
val bits = nextInt(2)
val halfn = _n >>> 1
val nextn =
if ((bits & 2) == 0) halfn
else _n - halfn
if ((bits & 1) == 0)
offset += _n - nextn
_n = nextn
}
offset + nextInt(_n.toInt)
}

/** Returns a pseudorandom, uniformly distributed long value between min
* (inclusive) and the specified value max (exclusive), drawn from this
* random number generator's sequence.
*/
def between(minInclusive: Long, maxExclusive: Long): Long = {
require(minInclusive < maxExclusive, "Invalid bounds")

val difference = maxExclusive - minInclusive
if (difference >= 0) {
nextLong(difference) + minInclusive
} else {
/* The interval size here is greater than Long.MaxValue,
* so the loop will exit with a probability of at least 1/2.
*/
def loop(): Long = {
val n = nextLong()
if (n >= minInclusive && n < maxExclusive) n
else loop()
}
loop()
}
}

/** Returns a pseudorandomly generated String. This routine does
* not take any measures to preserve the randomness of the distribution
* in the face of factors like unicode's variable-length encoding,
Expand Down
39 changes: 39 additions & 0 deletions test/junit/scala/util/RandomUtilTest.scala
Original file line number Diff line number Diff line change
@@ -0,0 +1,39 @@
package scala.util

import org.junit.{ Assert, Test }
import scala.util.Random

class RandomUtilTest {
@Test def testBetween: Unit = {
val rand = new Random()
Assert.assertTrue("Random between Int must be inclusive-exclusive", rand.between(0, 1).equals(0))
Assert.assertTrue("Random between Long must be inclusive-exclusive", rand.between(0L, 1L).equals(0L))
Assert.assertTrue("Random nextLong must be inclusive-exclusive", rand.nextLong(1L).equals(0L))
val float: Float = rand.between(0.0f, 1.0f)
Assert.assertTrue("Float Random should be inclusive-exclusive", 0.0f <= float && float < 1.0f)
val double: Double = rand.between(0.0f.toDouble, 1.0f.toDouble)
Assert.assertTrue("Random between Double should be inclusive-exclusive", 0.0f.toDouble <= double && double < 1.0f.toDouble)
}

@Test(expected = classOf[IllegalArgumentException]) def testBetweenIntException: Unit = {
val rand = new Random()
rand.between(1, 0)
}


@Test(expected = classOf[IllegalArgumentException]) def testBetweenLongException: Unit = {
val rand = new Random()
rand.between(1L, 0L)
}

@Test(expected = classOf[IllegalArgumentException]) def testBetweenFloatException: Unit = {
val rand = new Random()
rand.between(1.0f, 0.0f)
}

@Test(expected = classOf[IllegalArgumentException]) def testBetweenDoubleException: Unit = {
val rand = new Random()
rand.between(1.0f.toDouble, 0.0f.toDouble)
}

}

0 comments on commit f9a019c

Please sign in to comment.