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

com.github.tommyettinger.random.LaserRandom Maven / Gradle / Ivy

There is a newer version: 0.6.3
Show newest version
/*
 * Copyright (c) 2022-2023 See AUTHORS file.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *   http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 *
 */

package com.github.tommyettinger.random;

import java.util.Random;

/**
 * A faster and much-higher-quality substitute for {@link Random}. This allows many different random number
 * streams that don't overlap, and offers a more substantial API for commonly-used functions. This is not a
 * cryptographic random number generator, and should not be used in place of one.
 * 
* This fills in much of the functionality of MathUtils in libGDX, though with all code as instance methods * instead of static methods, and some things renamed (randomTriangular() became {@link #nextTriangular()}, * for instance, and random() became {@link #nextFloat()}). It also supplies some rare and sometimes-useful * code: {@link #skip(long)} allows "fast-forward" and "rewind" along with {@link #previousLong()} to simply * go back one step, you can get and set the exact state with {@link #getStateA()}, {@link #getStateB()}, * {@link #setStateA(long)}, {@link #setStateB(long)}, and {@link #setState(long, long)} (which is useful * if you want to save a LaserRandom and reload it later), and there's bounded int and long generators which * can use a negative number for their exclusive outer bound ({@link #nextSignedInt(int)} and * {@link #nextSignedLong(long)}, plus overloads that take an inner bound). There's float and double * generators that are inclusive on both ends ({@link #nextInclusiveFloat()}, and * {@link #nextInclusiveDouble()}. There's {@link #nextGaussian()}, which is implemented differently from * java.util.Random and always advances the state once. This implements all optional methods in * EnhancedRandom, and implements almost all EnhancedRandom methods explicitly, which allows LaserRandom to * be copied more easily without depending on juniper (see below). *
* Every method defined in this class advances the state by the same amount unless otherwise documented (only * {@link #nextTriangular()} and {@link #nextTriangular(float)} advance the state twice). The state can * advance 2 to the 64 times before the sequence of random numbers repeats, which would take a few years of * continuous generation. There are also 2 to the 63 possible sequences this can produce; you can tell which * one you're using with {@link #getStream()}. Note, {@link Random} can only advance 2 to the 48 times, which * takes under half a day to make it repeat on recent laptop hardware while also analyzing the numbers for * statistical issues. This generator is more comparable to SplittableRandom, introduced in JDK 8 but not * available in Android (even with desugaring) or GWT currently. SplittableRandom also can produce 2 to the * 64 numbers before repeating the sequence, and also has 2 to the 63 streams, but it will always produce * each possible long value exactly once over the course of that sequence. Each of LaserRandom's streams * produces a different sequence of numbers with a different set of numbers it omits and a different set of * numbers it produces more than once; each of SplittableRandom's streams simply rearranges the order of all * possible longs. Though it might seem like an issue that a LaserRandom stream has gaps in its possible * output, if you appended all 2 to the 63 possible LaserRandom streams in full, the gargantuan result would * include all longs equally often. So, if the stream is selected effectively at random, then the subset of * that stream that actually gets used should be fair (and it's very unlikely that any usage will need a full * stream of over 18 quintillion pseudo-random longs). It is strongly recommended that you use very different * numbers when creating many LaserRandom objects with similar states, because there is a noticeable * correlation between, for instance, a grid of LaserRandom objects initialized with stateA drawn from the * odd numbers 1 through 101, and stateB drawn from another odd number 1 through 101. Using * {@link #setSeed(long)} essentially eliminates this risk, so it's a good idea to seed this with one long. *
* If statistical quality is a concern, don't use {@link Random}, since the aforementioned * analysis finds statistical failures in about a minute when checking about 16GB of output; this class can * produce 64TB of random output without a tool like PractRand finding any failures (sometimes it can't find * any minor anomaly over several days of testing). RandomXS128 has some flaws, though they are not nearly as * severe as Random's; mostly they are limited to a particular kind of failure affecting the least * significant bits (the technical name for the test it fails is a "binary matrix rank" test, which a wide * variety of related generators can fail if they don't adequately randomize their outputs). RandomXS128's * flaws would be permissible if it was faster than any competitors, but it isn't, and there have been two * improved relatives of its algorithm published since it was created. Both of these improvements, * xoroshiro128** and xoshiro256**, are slower when implemented in Java than LaserRandom (also when all are * implemented in C and compiled with GCC or Clang, typically). There are also some concerns about specific * failure cases when the output of xoroshiro128** or xoshiro256** is multiplied by any of quadrillions of * constants and tested after that multiplication (see M.E. O'Neill's dissection of xoshiro256** * here). Xoshiro256**, like * LaserRandom, can't be reliably initialized using nearby values for its state variables, and does much * better if you use its setSeed(long) method. We do implement Xoshiro256** here, because it provides * 4-dimensional equidistribution, and that is hard to find. *
* You can copy this class independently of the library it's part of; it's meant as a general replacement for * Random and also RandomXS128. LaserRandom is generally faster than RandomXS128, and can be over 3x faster * when running on OpenJ9 (generating over 3 billion random long values per second). If you copy this, the * only step you probably need to do is to remove {@code extends EnhancedRandom} from the class, since * almost all of EnhancedRandom consists of either parent methods that this overrides explicitly, or to * provide a common interface for pseudo-random number generators on the JVM. This class avoids using the * Override annotation specifically because copying the class and removing the EnhancedRandom implementation * would cause compile errors if Override annotations were present. If you do keep this class implementing * EnhancedRandom, then that permits some extra methods to come in via default implementations, like * nextExclusiveFloat() (which uses the BitConversion class here from digital), minIntOf(), maxLongOf(), etc. *
* You may want to compare this class with TricycleRandom and FourWheelRandom in the same package; both of * those have a larger state size (and should usually have a larger period), are usually faster, and also * implement all of EnhancedRandom (except for {@link #skip(long)}), but they do even less randomizing for * the first result they return, so if the seeding has a pattern, then the start of their sequences will * have patterns. These patterns are less obvious but do persist in LaserRandom, and don't persist in * TricycleRandom or FourWheelRandom over a long run. All generators here do well when using * {@link #setSeed(long)} to set the full state. *
* Pew pew! Lasers! * * @author Tommy Ettinger */ public class LaserRandom extends EnhancedRandom { /** * Can be any long value. */ protected long stateA; /** * Must be odd. */ protected long stateB; /** * Creates a new LaserRandom. This constructor sets the states of the * random number generator to values very likely to be distinct from * any other invocation of this constructor. */ public LaserRandom () { super(); stateA = (long)((Math.random() - 0.5) * 0x1p52) ^ (long)((Math.random() - 0.5) * 0x1p64); stateB = (long)((Math.random() - 0.5) * 0x1p52) ^ (long)((Math.random() - 0.5) * 0x1p64) | 1L; } /** * Creates a new LaserRandom using a single {@code long} seed; the stream depends on whether the seed is even or odd. * * @param seed the initial seed * @see #setSeed(long) */ public LaserRandom (long seed) { super(seed); stateA = seed; stateB = seed | 1L; } /** * Creates a new LaserRandom using {@code seedA} exactly to set stateA (as with {@link #setStateA(long)},, * and using {@code seedB} to set stateB as with {@link #setStateB(long)} (meaning seedB will be used exactly if odd, * otherwise it will have 1 added to it and then used). * * @param seedA any long; will be used exactly to set stateA as with {@link #setStateA(long)} * @param seedB any odd long will be used exactly to set stateB, otherwise, as with {@link #setStateB(long)}, it will be made odd */ public LaserRandom (final long seedA, final long seedB) { super(seedA); stateA = seedA; stateB = seedB | 1L; } @Override public String getTag() { return "LasR"; } /** * LaserRandom has two possible states, both {@code long}. * The second state (selection {@code 1}) is always an odd number, and if * anything tries to set an even number to that state, the actual state used * will be one greater. * * @return 2 (two) */ public int getStateCount () { return 2; } /** * Get the "A" part of the internal state as a long. * * @return the current internal "A" state of this object. */ public long getStateA () { return stateA; } /** * Set the "A" part of the internal state with a long. * * @param stateA a 64-bit long */ public void setStateA (long stateA) { this.stateA = stateA; } /** * Get the "B" part of the internal state as a long. * * @return the current internal "B" state of this object. */ public long getStateB () { return stateB; } /** * Set the "B" part of the internal state with a long; the least significant bit is ignored (will always be odd). * That is, if stateB is odd, this uses it verbatim; if stateB is even, it adds 1 to it to make it odd. * * @param stateB a 64-bit long; the lowest bit will be ignored and the result always used as an odd number */ public void setStateB (long stateB) { this.stateB = stateB | 1L; } /** * Sets both parts of the internal state with one call; {@code stateA} is used verbatim, but {@code stateB} has * its least significant bit ignored and always overwritten with a '1' bit (meaning stateB will always be odd). * You can use any long for stateA without it being changed, and can use any odd long for stateB without it * being changed; as such, keeping {@code stateB} an odd number should be optimal. * * @param stateA a 64-bit long * @param stateB a 64-bit long; the lowest bit will be ignored and the result always used as an odd number */ public void setState (long stateA, long stateB) { this.stateA = stateA; this.stateB = stateB | 1L; } /** * Gets a selected state value from this LaserRandom. If selection is an even number, * this returns stateA, and if selection is odd, it returns stateB. This returns the * exact value of the selected state. * * @param selection used to select which state variable to get (usually 0 or 1) * @return the exact value of the selected state */ public long getSelectedState (int selection) { return (selection & 1) == 0 ? stateA : stateB; } /** * Sets a selected state value to the given long {@code value}. If selection is an even * number, this sets stateA to value as-is, and if selection is odd, this sets stateB to * value made odd (that is, if value is even, it uses value + 1, otherwise it uses value). * * @param selection used to select which state variable to set (usually 0 or 1) * @param value the exact value to use for the selected state, if valid */ public void setSelectedState (int selection, long value) { if ((selection & 1) == 0) stateA = value; else stateB = value | 1L; } /** * Sets the seed of this random number generator using a single * {@code long} seed. The general contract of {@code setSeed} is * that it alters the state of this random number generator object * so as to be in exactly the same state as if it had just been * created with the argument {@code seed} as a seed. * *

The implementation of {@code setSeed} by class * {@code LaserRandom} uses all 64 bits of the given seed for * {@link #setStateA(long)}, and all but the least-significant bit * of the seed for {@link #setStateB(long)} (the omitted bit is * always set to 1 in stateB, meaning stateB is always odd). * * @param seed the initial seed */ public void setSeed (long seed) { stateB = (stateA = seed) | 1L; } /** * Generates the next pseudorandom number with a specific maximum size in bits (not a max number). * If you want to get a random number in a range, you should usually use {@link #nextInt(int)} instead. * For some specific cases, this method is more efficient and less biased than {@link #nextInt(int)}. * For {@code bits} values between 1 and 30, this should be similar in effect to * {@code nextInt(1 << bits)}; though it won't typically produce the same values, they will have * the correct range. If {@code bits} is 31, this can return any non-negative {@code int}; note that * {@code nextInt(1 << 31)} won't behave this way because {@code 1 << 31} is negative. If * {@code bits} is 32 (or 0), this can return any {@code int}. * *

The general contract of {@code next} is that it returns an * {@code int} value and if the argument {@code bits} is between * {@code 1} and {@code 32} (inclusive), then that many low-order * bits of the returned value will be (approximately) independently * chosen bit values, each of which is (approximately) equally * likely to be {@code 0} or {@code 1}. *

* * @param bits the amount of random bits to request, from 1 to 32 * @return the next pseudorandom value from this random number * generator's sequence */ public int next (int bits) { final long s = stateA += 0xC6BC279692B5C323L; final long z = (s ^ s >>> 31) * (stateB += 0x9E3779B97F4A7C16L); return (int)(z ^ z >>> 26 ^ z >>> 6) >>> 32 - bits; } /** * Generates random bytes and places them into a user-supplied * byte array. The number of random bytes produced is equal to * the length of the byte array. * * @param bytes the byte array to fill with random bytes * @throws NullPointerException if the byte array is null */ public void nextBytes (byte[] bytes) { for (int i = 0; i < bytes.length; ) {for (long r = nextLong(), n = Math.min(bytes.length - i, 8); n-- > 0; r >>>= 8) {bytes[i++] = (byte)r;}} } /** * Returns the next pseudorandom, uniformly distributed {@code int} * value from this random number generator's sequence. The general * contract of {@code nextInt} is that one {@code int} value is * pseudorandomly generated and returned. All 232 possible * {@code int} values are produced with (approximately) equal probability. * * @return the next pseudorandom, uniformly distributed {@code int} * value from this random number generator's sequence */ public int nextInt () { final long s = stateA += 0xC6BC279692B5C323L; final long z = (s ^ s >>> 31) * (stateB += 0x9E3779B97F4A7C16L); return (int)(z ^ z >>> 26 ^ z >>> 6); } /** * Returns a pseudorandom, uniformly distributed {@code int} value * between 0 (inclusive) and the specified value (exclusive), drawn from * this random number generator's sequence. The general contract of * {@code nextInt} is that one {@code int} value in the specified range * is pseudorandomly generated and returned. All {@code bound} possible * {@code int} values are produced with (approximately) equal * probability. *
* It should be mentioned that the technique this uses has some bias, depending * on {@code bound}, but it typically isn't measurable without specifically looking * for it. Using the method this does allows this method to always advance the state * by one step, instead of a varying and unpredictable amount with the more typical * ways of rejection-sampling random numbers and only using numbers that can produce * an int within the bound without bias. * See M.E. O'Neill's * blog about random numbers for discussion of alternative, unbiased methods. * * @param bound the upper bound (exclusive). If negative or 0, this always returns 0. * @return the next pseudorandom, uniformly distributed {@code int} * value between zero (inclusive) and {@code bound} (exclusive) * from this random number generator's sequence */ public int nextInt (int bound) { final long s = stateA += 0xC6BC279692B5C323L; final long z = (s ^ s >>> 31) * (stateB += 0x9E3779B97F4A7C16L); return (int)(bound * ((z ^ z >>> 26 ^ z >>> 6) & 0xFFFFFFFFL) >> 32) & ~(bound >> 31); } /** * Returns a pseudorandom, uniformly distributed {@code int} value between an * inner bound of 0 (inclusive) and the specified {@code outerBound} (exclusive). * This is meant for cases where the outer bound may be negative, especially if * the bound is unknown or may be user-specified. A negative outer bound is used * as the lower bound; a positive outer bound is used as the upper bound. An outer * bound of -1, 0, or 1 will always return 0, keeping the bound exclusive (except * for outer bound 0). This method is slightly slower than {@link #nextInt(int)}. * * @param outerBound the outer exclusive bound; may be any int value, allowing negative * @return a pseudorandom int between 0 (inclusive) and outerBound (exclusive) * @see #nextInt(int) Here's a note about the bias present in the bounded generation. */ public int nextSignedInt (int outerBound) { final long s = stateA += 0xC6BC279692B5C323L; final long z = (s ^ s >>> 31) * (stateB += 0x9E3779B97F4A7C16L); outerBound = (int)(outerBound * ((z ^ z >>> 26 ^ z >>> 6) & 0xFFFFFFFFL) >> 32); return outerBound + (outerBound >>> 31); } /** * Returns a pseudorandom, uniformly distributed {@code int} value between the * specified {@code innerBound} (inclusive) and the specified {@code outerBound} * (exclusive). If {@code outerBound} is less than or equal to {@code innerBound}, * this always returns {@code innerBound}. Internally, this calls * {@link #nextLong(long, long)} and casts it to int (because the range between * innerBound and outerBound can be greater than the largest int). * *
For any case where outerBound might be valid but less than innerBound, you * can use {@link #nextSignedInt(int, int)}. * * @param innerBound the inclusive inner bound; may be any int, allowing negative * @param outerBound the exclusive outer bound; must be greater than innerBound (otherwise this returns innerBound) * @return a pseudorandom int between innerBound (inclusive) and outerBound (exclusive) * @see #nextInt(int) Here's a note about the bias present in the bounded generation. */ public int nextInt (int innerBound, int outerBound) { return (int)nextLong(innerBound, outerBound); } /** * Returns a pseudorandom, uniformly distributed {@code int} value between the * specified {@code innerBound} (inclusive) and the specified {@code outerBound} * (exclusive). This is meant for cases where either bound may be negative, * especially if the bounds are unknown or may be user-specified. It is slightly * slower than {@link #nextInt(int, int)}. Internally, this calls * {@link #nextSignedLong(long, long)} and casts it to int (because the range * between innerBound and outerBound can be greater than the largest int). * * @param innerBound the inclusive inner bound; may be any int, allowing negative * @param outerBound the exclusive outer bound; may be any int, allowing negative * @return a pseudorandom int between innerBound (inclusive) and outerBound (exclusive) * @see #nextInt(int) Here's a note about the bias present in the bounded generation. */ public int nextSignedInt (int innerBound, int outerBound) { return (int)nextSignedLong(innerBound, outerBound); } /** * Returns the next pseudorandom, uniformly distributed {@code long} * value from this random number generator's sequence. The general * contract of {@code nextLong} is that one {@code long} value is * pseudorandomly generated and returned. *
* An individual {@code LaserRNG} can't return all 18-quintillion possible {@code long} values, * but the full set of 9-quintillion possible random number streams that this class can produce will, * as a whole, produce all {@code long} values with equal likelihood. * * @return the next pseudorandom, uniformly distributed {@code long} * value from this random number generator's sequence */ public long nextLong () { final long s = stateA += 0xC6BC279692B5C323L; final long z = (s ^ s >>> 31) * (stateB += 0x9E3779B97F4A7C16L); return z ^ z >>> 26 ^ z >>> 6; } /** * Returns a pseudorandom, uniformly distributed {@code long} value * between 0 (inclusive) and the specified value (exclusive), drawn from * this random number generator's sequence. The general contract of * {@code nextLong} is that one {@code long} value in the specified range * is pseudorandomly generated and returned. All {@code bound} possible * {@code long} values are produced with (approximately) equal * probability, though there is a small amount of bias depending on the bound. * *
Note that this advances the state by the same amount as a single call to * {@link #nextLong()}, which allows methods like {@link #skip(long)} to function * correctly, but introduces some bias when {@code bound} is very large. This will * also advance the state if {@code bound} is 0 or negative, so usage with a variable * bound will advance the state reliably. * *
This method has some bias, particularly on larger bounds. Actually measuring * bias with bounds in the trillions or greater is challenging but not impossible, so * don't use this for a real-money gambling purpose. The bias isn't especially * significant, though. * * @param bound the upper bound (exclusive). If negative or 0, this always returns 0. * @return the next pseudorandom, uniformly distributed {@code long} * value between zero (inclusive) and {@code bound} (exclusive) * from this random number generator's sequence * @see #nextInt(int) Here's a note about the bias present in the bounded generation. */ public long nextLong (long bound) { return nextLong(0L, bound); } /** * Returns a pseudorandom, uniformly distributed {@code long} value between an * inner bound of 0 (inclusive) and the specified {@code outerBound} (exclusive). * This is meant for cases where the outer bound may be negative, especially if * the bound is unknown or may be user-specified. A negative outer bound is used * as the lower bound; a positive outer bound is used as the upper bound. An outer * bound of -1, 0, or 1 will always return 0, keeping the bound exclusive (except * for outer bound 0). * *

Note that this advances the state by the same amount as a single call to * {@link #nextLong()}, which allows methods like {@link #skip(long)} to function * correctly, but introduces some bias when {@code bound} is very large. This * method should be about as fast as {@link #nextLong(long)} , unlike the speed * difference between {@link #nextInt(int)} and {@link #nextSignedInt(int)}. * * @param outerBound the outer exclusive bound; may be any long value, allowing negative * @return a pseudorandom long between 0 (inclusive) and outerBound (exclusive) * @see #nextInt(int) Here's a note about the bias present in the bounded generation. */ public long nextSignedLong (long outerBound) { return nextSignedLong(0L, outerBound); } /** * Returns a pseudorandom, uniformly distributed {@code long} value between the * specified {@code innerBound} (inclusive) and the specified {@code outerBound} * (exclusive). If {@code outerBound} is less than or equal to {@code innerBound}, * this always returns {@code innerBound}. * *
For any case where outerBound might be valid but less than innerBound, you * can use {@link #nextSignedLong(long, long)}. * * @param inner the inclusive inner bound; may be any long, allowing negative * @param outer the exclusive outer bound; must be greater than innerBound (otherwise this returns innerBound) * @return a pseudorandom long between innerBound (inclusive) and outerBound (exclusive) * @see #nextInt(int) Here's a note about the bias present in the bounded generation. */ public long nextLong (long inner, long outer) { final long rand = nextLong(); if (inner >= outer) return inner; final long bound = outer - inner; final long randLow = rand & 0xFFFFFFFFL; final long boundLow = bound & 0xFFFFFFFFL; final long randHigh = (rand >>> 32); final long boundHigh = (bound >>> 32); return inner + (randHigh * boundLow >>> 32) + (randLow * boundHigh >>> 32) + randHigh * boundHigh; } /** * Returns a pseudorandom, uniformly distributed {@code long} value between the * specified {@code innerBound} (inclusive) and the specified {@code outerBound} * (exclusive). This is meant for cases where either bound may be negative, * especially if the bounds are unknown or may be user-specified. * * @param inner the inclusive inner bound; may be any long, allowing negative * @param outer the exclusive outer bound; may be any long, allowing negative * @return a pseudorandom long between innerBound (inclusive) and outerBound (exclusive) * @see #nextInt(int) Here's a note about the bias present in the bounded generation. */ public long nextSignedLong (long inner, long outer) { final long rand = nextLong(); if (outer < inner) { long t = outer; outer = inner + 1L; inner = t + 1L; } final long bound = outer - inner; final long randLow = rand & 0xFFFFFFFFL; final long boundLow = bound & 0xFFFFFFFFL; final long randHigh = (rand >>> 32); final long boundHigh = (bound >>> 32); return inner + (randHigh * boundLow >>> 32) + (randLow * boundHigh >>> 32) + randHigh * boundHigh; } /** * Returns the next pseudorandom, uniformly distributed * {@code boolean} value from this random number generator's * sequence. The general contract of {@code nextBoolean} is that one * {@code boolean} value is pseudorandomly generated and returned. The * values {@code true} and {@code false} are produced with * (approximately) equal probability. * * @return the next pseudorandom, uniformly distributed * {@code boolean} value from this random number generator's * sequence */ public boolean nextBoolean () { final long s = stateA += 0xC6BC279692B5C323L; return (s ^ s >>> 31) * (stateB += 0x9E3779B97F4A7C16L) < 0L; } /** * Returns the next pseudorandom, uniformly distributed {@code float} * value between {@code 0.0} (inclusive) and {@code 1.0} (exclusive) * from this random number generator's sequence. * *

The general contract of {@code nextFloat} is that one * {@code float} value, chosen (approximately) uniformly from the * range {@code 0.0f} (inclusive) to {@code 1.0f} (exclusive), is * pseudorandomly generated and returned. All 224 possible * {@code float} values of the form m x 2-24, * where m is a positive integer less than 224, are * produced with (approximately) equal probability. * *

The hedge "approximately" is used in the foregoing description only * because the next method is only approximately an unbiased source of * independently chosen bits. If it were a perfect source of randomly * chosen bits, then the algorithm shown would choose {@code float} * values from the stated range with perfect uniformity.

* * @return the next pseudorandom, uniformly distributed {@code float} * value between {@code 0.0} and {@code 1.0} from this * random number generator's sequence */ public float nextFloat () { final long s = stateA += 0xC6BC279692B5C323L; final long z = (s ^ s >>> 31) * (stateB += 0x9E3779B97F4A7C16L); return ((z ^ z >>> 6) >>> 40) * 0x1p-24f; } /** * Gets a pseudo-random float between 0 (inclusive) and {@code outerBound} (exclusive). * The outerBound may be positive or negative. * Exactly the same as {@code nextFloat() * outerBound}. * * @param outerBound the exclusive outer bound * @return a float between 0 (inclusive) and {@code outerBound} (exclusive) */ public float nextFloat (float outerBound) { return nextFloat() * outerBound; } /** * Gets a pseudo-random float between {@code innerBound} (inclusive) and {@code outerBound} (exclusive). * Either, neither, or both of innerBound and outerBound may be negative; this does not change which is * inclusive and which is exclusive. * * @param innerBound the inclusive inner bound; may be negative * @param outerBound the exclusive outer bound; may be negative * @return a float between {@code innerBound} (inclusive) and {@code outerBound} (exclusive) */ public float nextFloat (float innerBound, float outerBound) { return innerBound + nextFloat() * (outerBound - innerBound); } /** * Returns the next pseudorandom, uniformly distributed * {@code double} value between {@code 0.0} (inclusive) and {@code 1.0} * (exclusive) from this random number generator's sequence. * *

The general contract of {@code nextDouble} is that one * {@code double} value, chosen (approximately) uniformly from the * range {@code 0.0d} (inclusive) to {@code 1.0d} (exclusive), is * pseudorandomly generated and returned. * *

The hedge "approximately" is used in the foregoing description only * because the {@code next} method is only approximately an unbiased * source of independently chosen bits. If it were a perfect source of * randomly chosen bits, then the algorithm shown would choose * {@code double} values from the stated range with perfect uniformity. * * @return the next pseudorandom, uniformly distributed {@code double} * value between {@code 0.0} and {@code 1.0} from this * random number generator's sequence */ public double nextDouble () { final long s = stateA += 0xC6BC279692B5C323L; final long z = (s ^ s >>> 31) * (stateB += 0x9E3779B97F4A7C16L); return (z >>> 11 ^ z >>> 37 ^ z >>> 17) * 0x1.0p-53; } /** * Gets a pseudo-random double between 0 (inclusive) and {@code outerBound} (exclusive). * The outerBound may be positive or negative. * Exactly the same as {@code nextDouble() * outerBound}. * * @param outerBound the exclusive outer bound * @return a double between 0 (inclusive) and {@code outerBound} (exclusive) */ public double nextDouble (double outerBound) { return nextDouble() * outerBound; } /** * Gets a pseudo-random double between {@code innerBound} (inclusive) and {@code outerBound} (exclusive). * Either, neither, or both of innerBound and outerBound may be negative; this does not change which is * inclusive and which is exclusive. * * @param innerBound the inclusive inner bound; may be negative * @param outerBound the exclusive outer bound; may be negative * @return a double between {@code innerBound} (inclusive) and {@code outerBound} (exclusive) */ public double nextDouble (double innerBound, double outerBound) { return innerBound + nextDouble() * (outerBound - innerBound); } /** * This is just like {@link #nextDouble()}, returning a double between 0 and 1, except that it is inclusive on both 0.0 and 1.0. * It returns 1.0 extremely rarely, 0.000000000000011102230246251565% of the time if there is no bias in the generator, but it * can happen. This uses {@link #nextLong(long)} internally, so it may have some bias towards or against specific * subtly-different results. Other generators here use BitConversion and a very different algorithm, but this avoids * BitConversion so that it can be copied more easily. * * @return a double between 0.0, inclusive, and 1.0, inclusive */ public double nextInclusiveDouble () { // return nextLong(0x20000000000001L) * 0x1p-53; final long rand = nextLong(); final long bound = 0x20000000000001L; final long randLow = rand & 0xFFFFFFFFL; final long randHigh = (rand >>> 32); final long boundHigh = (bound >>> 32); return ((randLow * boundHigh >>> 32) + randHigh * boundHigh) * 0x1p-53; } /** * Just like {@link #nextDouble(double)}, but this is inclusive on both 0.0 and {@code outerBound}. * It may be important to note that it returns outerBound on only 0.000000000000011102230246251565% of calls. * * @param outerBound the outer inclusive bound; may be positive or negative * @return a double between 0.0, inclusive, and {@code outerBound}, inclusive */ public double nextInclusiveDouble (double outerBound) { return nextInclusiveDouble() * outerBound; } /** * Just like {@link #nextDouble(double, double)}, but this is inclusive on both {@code innerBound} and {@code outerBound}. * It may be important to note that it returns outerBound on only 0.000000000000011102230246251565% of calls, if it can * return it at all because of floating-point imprecision when innerBound is a larger number. * * @param innerBound the inner inclusive bound; may be positive or negative * @param outerBound the outer inclusive bound; may be positive or negative * @return a double between {@code innerBound}, inclusive, and {@code outerBound}, inclusive */ public double nextInclusiveDouble (double innerBound, double outerBound) { return innerBound + nextInclusiveDouble() * (outerBound - innerBound); } /** * This is just like {@link #nextFloat()}, returning a float between 0 and 1, except that it is inclusive on both 0.0 and 1.0. * It returns 1.0 rarely, 0.00000596046412226771% of the time if there is no bias in the generator, but it can happen. This method * has been tested by generating 268435456 (or 0x10000000) random ints with {@link #nextInt(int)}, and just before the end of that * it had generated every one of the 16777217 roughly-equidistant floats this is able to produce. Not all seeds and streams are * likely to accomplish that in the same time, or at all, depending on the generator. * * @return a float between 0.0, inclusive, and 1.0, inclusive */ public float nextInclusiveFloat () { return (int)(0x1000001L * (nextLong() & 0xFFFFFFFFL) >> 32) * 0x1p-24f; } /** * Just like {@link #nextFloat(float)}, but this is inclusive on both 0.0 and {@code outerBound}. * It may be important to note that it returns outerBound on only 0.00000596046412226771% of calls. * * @param outerBound the outer inclusive bound; may be positive or negative * @return a float between 0.0, inclusive, and {@code outerBound}, inclusive */ public float nextInclusiveFloat (float outerBound) { return nextInclusiveFloat() * outerBound; } /** * Just like {@link #nextFloat(float, float)}, but this is inclusive on both {@code innerBound} and {@code outerBound}. * It may be important to note that it returns outerBound on only 0.00000596046412226771% of calls, if it can return * it at all because of floating-point imprecision when innerBound is a larger number. * * @param innerBound the inner inclusive bound; may be positive or negative * @param outerBound the outer inclusive bound; may be positive or negative * @return a float between {@code innerBound}, inclusive, and {@code outerBound}, inclusive */ public float nextInclusiveFloat (float innerBound, float outerBound) { return innerBound + nextInclusiveFloat() * (outerBound - innerBound); } /** * Returns the next pseudorandom, Gaussian ("normally") distributed * {@code double} value with mean {@code 0.0} and standard * deviation {@code 1.0} from this random number generator's sequence. *

* The general contract of {@code nextGaussian} is that one * {@code double} value, chosen from (approximately) the usual * normal distribution with mean {@code 0.0} and standard deviation * {@code 1.0}, is pseudorandomly generated and returned. *

* This uses an inverse transform sample method to generate its random * values, using a method called probit(). The algorithm used was * written by Peter John Acklam, and implemented by Sherali Karimov. * Original source. * Information on the algorithm. * As this is written, it can return a maximum result of 8.209536145151493, * and a minimum result of -38.5 . 38.5 only occurs 1 in every (2 to the 53) * calls, on average, and all other results are between -8.209536145151493 * and 8.209536145151493 . Note, this implementation is different from the * one in EnhancedRandom only to permit copying this class more easily; * EnhancedRandom normally uses another class, Ziggurat, to more efficiently * generate normally-distributed doubles. *
* @return the next pseudorandom, Gaussian ("normally") distributed * {@code double} value with mean {@code 0.0} and standard deviation * {@code 1.0} from this random number generator's sequence */ public double nextGaussian () { final double d = nextDouble(); if (d == 0) { return -38.5; } else if (d < 0.02425) { final double q = Math.sqrt(-2.0 * Math.log(d)); return (((((-7.784894002430293e-03 * q - 3.223964580411365e-01) * q - 2.400758277161838e+00) * q - 2.549732539343734e+00) * q + 4.374664141464968e+00) * q + 2.938163982698783e+00) / ( (((7.784695709041462e-03 * q + 3.224671290700398e-01) * q + 2.445134137142996e+00) * q + 3.754408661907416e+00) * q + 1.0); } else if (0.97575 < d) { final double q = Math.sqrt(-2.0 * Math.log(1 - d)); return -(((((-7.784894002430293e-03 * q - 3.223964580411365e-01) * q - 2.400758277161838e+00) * q - 2.549732539343734e+00) * q + 4.374664141464968e+00) * q + 2.938163982698783e+00) / ( (((7.784695709041462e-03 * q + 3.224671290700398e-01) * q + 2.445134137142996e+00) * q + 3.754408661907416e+00) * q + 1.0); } else { final double q = d - 0.5; final double r = q * q; return (((((-3.969683028665376e+01 * r + 2.209460984245205e+02) * r - 2.759285104469687e+02) * r + 1.383577518672690e+02) * r - 3.066479806614716e+01) * r + 2.506628277459239e+00) * q / ( ((((-5.447609879822406e+01 * r + 1.615858368580409e+02) * r - 1.556989798598866e+02) * r + 6.680131188771972e+01) * r - 1.328068155288572e+01) * r + 1.0); } } /** * Advances or rolls back the {@code LaserRandom}' state without actually generating each number. Skips forward * or backward a number of steps specified by advance, where a step is equal to one call to {@link #nextLong()}, * and returns the random number produced at that step. Negative numbers can be used to step backward, or 0 can be * given to get the most-recently-generated long from {@link #nextLong()}. * *

Note that none of the number-generating methods here advance state differently from {@link #nextLong()} except * for the Stream APIs. This is somewhat unusual; in many generators, calls to {@link #nextInt(int)} and similar * bounded-range random generators can advance the state by a variable amount. Using a fixed advance permits this * method and also allows guaranteeing the cycle length (also called period), but introduces a tiny amount of bias * for some bounds (mostly very large ones). * * @param advance Number of future generations to skip over; can be negative to backtrack, 0 gets the most-recently-generated number * @return the random long generated after skipping forward or backwards by {@code advance} numbers */ public long skip (long advance) { final long s = stateA += 0xC6BC279692B5C323L * advance; final long z = (s ^ s >>> 31) * (stateB += 0x9E3779B97F4A7C16L * advance); return z ^ z >>> 26 ^ z >>> 6; } public long previousLong () { final long s = stateA; final long z = (s ^ s >>> 31) * stateB; stateA -= 0xC6BC279692B5C323L; stateB -= 0x9E3779B97F4A7C16L; return z ^ z >>> 26 ^ z >>> 6; } /** * Creates a new {@code LaserRandom} with identical states to this one, so if the same LaserRandom methods are * called on this object and its copy (in the same order), the same outputs will be produced. This is not * guaranteed to copy the inherited state of the {@link Random} parent class, so if you call methods that are * only implemented by Random and not LaserRandom, the results may differ. * * @return a deep copy of this LaserRandom. */ public LaserRandom copy () { return new LaserRandom(stateA, stateB); } /** * Gets a long that identifies which stream of numbers this generator is producing; this stream identifier is always * an odd long and won't change by generating numbers. It is determined at construction and will usually (not * always) change if {@link #setStateA(long)} or {@link #setStateB(long)} are called. Each stream is a * probably-unique sequence of 2 to the 64 longs, where approximately 1/3 of all possible longs will not ever occur * (while others occur twice or more), but this set of results is different for every stream. There are 2 to the 63 * possible streams, one for every odd long. * * @return an odd long that identifies which stream this LaserRandom is generating from */ public long getStream () { return stateB - stateA * 0x3085776F0FBEB7F2L; // 0x3085776F0FBEB7F2L == 0x1743CE5C6E1B848BL * 0x9E3779B97F4A7C16L } /** * Returns true if a random value between 0 and 1 is less than the specified value. * * @param chance a float between 0.0 and 1.0; higher values are more likely to result in true * @return a boolean selected with the given {@code chance} of being true */ public boolean nextBoolean (float chance) { return nextFloat() < chance; } /** * Returns -1 or 1, randomly. * * @return -1 or 1, selected with approximately equal likelihood */ public int nextSign () { return 1 | nextInt() >> 31; } /** * Returns a triangularly distributed random number between -1.0 (exclusive) and 1.0 (exclusive), where values around zero are * more likely. Advances the state twice. *

* This is an optimized version of {@link #nextTriangular(float, float, float) randomTriangular(-1, 1, 0)} */ public float nextTriangular () { return nextFloat() - nextFloat(); } /** * Returns a triangularly distributed random number between {@code -max} (exclusive) and {@code max} (exclusive), where values * around zero are more likely. Advances the state twice. *

* This is an optimized version of {@link #nextTriangular(float, float, float) randomTriangular(-max, max, 0)} * * @param max the upper limit */ public float nextTriangular (float max) { return (nextFloat() - nextFloat()) * max; } /** * Returns a triangularly distributed random number between {@code min} (inclusive) and {@code max} (exclusive), where the * {@code mode} argument defaults to the midpoint between the bounds, giving a symmetric distribution. Advances the state once. *

* This method is equivalent of {@link #nextTriangular(float, float, float) randomTriangular(min, max, (min + max) * 0.5f)} * * @param min the lower limit * @param max the upper limit */ public float nextTriangular (float min, float max) { return nextTriangular(min, max, (min + max) * 0.5f); } /** * Returns a triangularly distributed random number between {@code min} (inclusive) and {@code max} (exclusive), where values * around {@code mode} are more likely. Advances the state once. * * @param min the lower limit * @param max the upper limit * @param mode the point around which the values are more likely */ public float nextTriangular (float min, float max, float mode) { float u = nextFloat(); float d = max - min; if (u <= (mode - min) / d) {return min + (float)Math.sqrt(u * d * (mode - min));} return max - (float)Math.sqrt((1 - u) * d * (max - mode)); } /** * Gets a randomly-selected item from the given array, which must be non-null and non-empty * * @param array a non-null, non-empty array of {@code T} items * @param any reference type * @return a random item from {@code array} * @throws NullPointerException if array is null * @throws IndexOutOfBoundsException if array is empty */ public T randomElement (T[] array) { return array[nextInt(array.length)]; } /** * Shuffles the given array in-place pseudo-randomly, using this to determine how to shuffle. * * @param items an int array; must be non-null */ public void shuffle (int[] items) { for (int i = items.length - 1; i > 0; i--) { int ii = nextInt(i + 1); int temp = items[i]; items[i] = items[ii]; items[ii] = temp; } } /** * Shuffles the given array in-place pseudo-randomly, using this to determine how to shuffle. * * @param items a long array; must be non-null */ public void shuffle (long[] items) { for (int i = items.length - 1; i > 0; i--) { int ii = nextInt(i + 1); long temp = items[i]; items[i] = items[ii]; items[ii] = temp; } } /** * Shuffles the given array in-place pseudo-randomly, using this to determine how to shuffle. * * @param items a float array; must be non-null */ public void shuffle (float[] items) { for (int i = items.length - 1; i > 0; i--) { int ii = nextInt(i + 1); float temp = items[i]; items[i] = items[ii]; items[ii] = temp; } } /** * Shuffles the given array in-place pseudo-randomly, using this to determine how to shuffle. * * @param items a char array; must be non-null */ public void shuffle (char[] items) { for (int i = items.length - 1; i > 0; i--) { int ii = nextInt(i + 1); char temp = items[i]; items[i] = items[ii]; items[ii] = temp; } } /** * Shuffles the given array in-place pseudo-randomly, using this to determine how to shuffle. * * @param items a byte array; must be non-null */ public void shuffle (byte[] items) { for (int i = items.length - 1; i > 0; i--) { int ii = nextInt(i + 1); byte temp = items[i]; items[i] = items[ii]; items[ii] = temp; } } /** * Shuffles the given array in-place pseudo-randomly, using this to determine how to shuffle. * * @param items a double array; must be non-null */ public void shuffle (double[] items) { for (int i = items.length - 1; i > 0; i--) { int ii = nextInt(i + 1); double temp = items[i]; items[i] = items[ii]; items[ii] = temp; } } /** * Shuffles the given array in-place pseudo-randomly, using this to determine how to shuffle. * * @param items a short array; must be non-null */ public void shuffle (short[] items) { for (int i = items.length - 1; i > 0; i--) { int ii = nextInt(i + 1); short temp = items[i]; items[i] = items[ii]; items[ii] = temp; } } /** * Shuffles the given array in-place pseudo-randomly, using this to determine how to shuffle. * * @param items a boolean array; must be non-null */ public void shuffle (boolean[] items) { for (int i = items.length - 1; i > 0; i--) { int ii = nextInt(i + 1); boolean temp = items[i]; items[i] = items[ii]; items[ii] = temp; } } /** * Shuffles the given array in-place pseudo-randomly, using this to determine how to shuffle. * * @param items an array of some reference type; must be non-null but may contain null items */ public void shuffle (T[] items) { for (int i = items.length - 1; i > 0; i--) { int ii = nextInt(i + 1); T temp = items[i]; items[i] = items[ii]; items[ii] = temp; } } @Override public boolean equals (Object o) { if (this == o) return true; if (o == null || getClass() != o.getClass()) return false; LaserRandom that = (LaserRandom)o; if (stateA != that.stateA) return false; return stateB == that.stateB; } /** * This String conversion uses base-10 numbers for the states, unlike all other EnhancedRandom implementations, which use base-16. * This is done here to avoid a dependency on Base, allowing this class to be copied more easily. * @return a String description of this LaserRandom and its two states */ @Override public String toString () { return "LaserRandom{" + "stateA=" + stateA + "L, stateB=" + stateB + "L}"; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy