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

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

/*
 * 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 com.github.tommyettinger.digital.Base;
import com.github.tommyettinger.digital.BitConversion;
import com.github.tommyettinger.digital.Distributor;
import com.github.tommyettinger.digital.MathTools;

import java.io.Externalizable;
import java.io.IOException;
import java.io.ObjectInput;
import java.io.ObjectOutput;
import java.util.List;
import java.util.Random;

/**
 * A superset of the functionality in {@link java.util.Random}, meant for random number generators
 * that would be too bare-bones with just Random's methods.
 */
public abstract class EnhancedRandom extends Random implements Externalizable {

	public EnhancedRandom() {
		super();
	}

	public EnhancedRandom(long seed) {
		super(seed);
	}

	/**
	 * Gets the tag used to identify this type of EnhancedRandom, as a String. This tag should be unique,
	 * and for uniformity purposes, all tags used in this library are 4 characters long. User-defined tags
	 * should have a different length.
	 * @return a unique String identifier for this type of EnhancedRandom; usually 4 chars long.
	 */
	public abstract String getTag();

	/**
	 * Uses {@link Math#random()} to hastily put together a not-especially-uniform {@code long} value,
	 * meant only to produce a seed when no seed was specified (the "I don't care" seed).
	 *
	 * @return a kind-of-uniform {@code long} value
	 */
	public static long seedFromMath () {
		return (long)((Math.random() - 0.5) * 0x1p52) ^ (long)((Math.random() - 0.5) * 0x1p64);
	}

	/**
	 * Sets the seed of this random number generator using a single
	 * {@code long} seed. This should behave exactly the same as if a new
	 * object of this type was created with the constructor that takes a single
	 * {@code long} value. This does not necessarily assign the state
	 * variable(s) of the implementation with the exact contents of seed, so
	 * {@link #getSelectedState(int)} should not be expected to return
	 * {@code seed} after this, though it may. If this implementation has more
	 * than one {@code long} of state, then the expectation is that none of
	 * those state variables will be exactly equal to {@code seed} (almost all
	 * the time).
	 *
	 * @param seed the initial seed
	 */
	public abstract void setSeed (long seed);

	/**
	 * Gets the number of possible state variables that can be selected with
	 * {@link #getSelectedState(int)} or {@link #setSelectedState(int, long)}.
	 * This defaults to returning 0, making no state variable available for
	 * reading or writing. An implementation that has only one {@code long}
	 * state, like {@link DistinctRandom} generator, should return {@code 1}. A
	 * generator that permits setting two different {@code long} values, like
	 * {@link LaserRandom}, should return {@code 2}. Much larger values are
	 * possible for types like the Mersenne Twister or some CMWC generators.
	 *
	 * @return the non-negative number of selections possible for state variables
	 */
	public int getStateCount () {
		return 0;
	}

	/**
	 * Gets a selected state value from this EnhancedRandom. The number of possible selections
	 * is up to the implementing class, and is accessible via {@link #getStateCount()}, but
	 * negative values for {@code selection} are typically not tolerated. This should return
	 * the exact value of the selected state, assuming it is implemented. The default
	 * implementation throws an UnsupportedOperationException, and implementors only have to
	 * allow reading the state if they choose to implement this differently. If this method
	 * is intended to be used, {@link #getStateCount()} must also be implemented.
	 *
	 * @param selection used to select which state variable to get; generally non-negative
	 * @return the exact value of the selected state
	 */
	public long getSelectedState (int selection) {
		throw new UnsupportedOperationException("getSelectedState() not supported.");
	}

	/**
	 * Sets a selected state value to the given long {@code value}. The number of possible
	 * selections is up to the implementing class, but negative values for {@code selection}
	 * are typically not tolerated. Implementors are permitted to change {@code value} if it
	 * is not valid, but they should not alter it if it is valid. The public implementation
	 * calls {@link #setSeed(long)} with {@code value}, which doesn't need changing if the
	 * generator has one state that is set verbatim by setSeed(). Otherwise, this method
	 * should be implemented when {@link #getSelectedState(int)} is and the state is allowed
	 * to be set by users. Having accurate ways to get and set the full state of a random
	 * number generator makes it much easier to serialize and deserialize that class.
	 *
	 * @param selection used to select which state variable to set; generally non-negative
	 * @param value     the exact value to use for the selected state, if valid
	 */
	public void setSelectedState (int selection, long value) {
		setSeed(value);
	}

	/**
	 * Sets each state variable to the given {@code state}. If {@link #getStateCount()} is
	 * 1, then this should set the whole state to the given value using
	 * {@link #setSelectedState(int, long)}. If getStateCount() is more than 1, then all
	 * states will be set in the same way (using setSelectedState(), all to {@code state}).
	 *
	 * @param state the long value to use for each state variable
	 */
	public void setState (long state) {
		for (int i = getStateCount() - 1; i >= 0; i--) {
			setSelectedState(i, state);
		}
	}

	/**
	 * Sets each state variable to either {@code stateA} or {@code stateB}, alternating.
	 * This uses {@link #setSelectedState(int, long)} to set the values. If there is one
	 * state variable ({@link #getStateCount()} is 1), then this only sets that state
	 * variable to stateA. If there are two state variables, the first is set to stateA,
	 * and the second to stateB. If there are more, it reuses stateA, then stateB, then
	 * stateA, and so on until all variables are set.
	 *
	 * @param stateA the long value to use for states at index 0, 2, 4, 6...
	 * @param stateB the long value to use for states at index 1, 3, 5, 7...
	 */
	public void setState (long stateA, long stateB) {
		final int c = getStateCount();
		for (int i = 0; i < c; i += 2) {
			setSelectedState(i, stateA);
		}
		for (int i = 1; i < c; i += 2) {
			setSelectedState(i, stateB);
		}
	}

	/**
	 * Sets each state variable to {@code stateA}, {@code stateB}, or {@code stateC},
	 * alternating. This uses {@link #setSelectedState(int, long)} to set the values.
	 * If there is one state variable ({@link #getStateCount()} is 1), then this only
	 * sets that state variable to stateA. If there are two state variables, the first
	 * is set to stateA, and the second to stateB. With three state variables, the
	 * first is set to stateA, the second to stateB, and the third to stateC. If there
	 * are more, it reuses stateA, then stateB, then stateC, then stateA, and so on
	 * until all variables are set.
	 *
	 * @param stateA the long value to use for states at index 0, 3, 6, 9...
	 * @param stateB the long value to use for states at index 1, 4, 7, 10...
	 * @param stateC the long value to use for states at index 2, 5, 8, 11...
	 */
	public void setState (long stateA, long stateB, long stateC) {
		final int c = getStateCount();
		for (int i = 0; i < c; i += 3) {
			setSelectedState(i, stateA);
		}
		for (int i = 1; i < c; i += 3) {
			setSelectedState(i, stateB);
		}
		for (int i = 2; i < c; i += 3) {
			setSelectedState(i, stateC);
		}
	}

	/**
	 * Sets each state variable to {@code stateA}, {@code stateB}, {@code stateC}, or
	 * {@code stateD}, alternating. This uses {@link #setSelectedState(int, long)} to
	 * set the values. If there is one state variable ({@link #getStateCount()} is 1),
	 * then this only sets that state variable to stateA. If there are two state
	 * variables, the first is set to stateA, and the second to stateB. With three
	 * state variables, the first is set to stateA, the second to stateB, and the third
	 * to stateC. With four state variables, the first is set to stateA, the second to
	 * stateB, the third to stateC, and the fourth to stateD. If there are more, it
	 * reuses stateA, then stateB, then stateC, then stateD, then stateA, and so on
	 * until all variables are set.
	 *
	 * @param stateA the long value to use for states at index 0, 4, 8, 12...
	 * @param stateB the long value to use for states at index 1, 5, 9, 13...
	 * @param stateC the long value to use for states at index 2, 6, 10, 14...
	 * @param stateD the long value to use for states at index 3, 7, 11, 15...
	 */
	public void setState (long stateA, long stateB, long stateC, long stateD) {
		final int c = getStateCount();
		for (int i = 0; i < c; i += 4) {
			setSelectedState(i, stateA);
		}
		for (int i = 1; i < c; i += 4) {
			setSelectedState(i, stateB);
		}
		for (int i = 2; i < c; i += 4) {
			setSelectedState(i, stateC);
		}
		for (int i = 3; i < c; i += 4) {
			setSelectedState(i, stateD);
		}
	}

	/**
	 * Sets each state variable to {@code stateA}, {@code stateB}, {@code stateC}, or
	 * {@code stateD}, alternating. This uses {@link #setSelectedState(int, long)} to
	 * set the values. If there is one state variable ({@link #getStateCount()} is 1),
	 * then this only sets that state variable to stateA. If there are two state
	 * variables, the first is set to stateA, and the second to stateB. With three
	 * state variables, the first is set to stateA, the second to stateB, and the third
	 * to stateC. With four state variables, the first is set to stateA, the second to
	 * stateB, the third to stateC, and the fourth to stateD. If there are more, it
	 * reuses stateA, then stateB, then stateC, then stateD, then stateA, and so on
	 * until all variables are set.
	 *
	 * @param stateA the long value to use for states at index 0, 5, 10, 15...
	 * @param stateB the long value to use for states at index 1, 6, 11, 16...
	 * @param stateC the long value to use for states at index 2, 7, 12, 17...
	 * @param stateD the long value to use for states at index 3, 8, 13, 18...
	 * @param stateE the long value to use for states at index 4, 9, 14, 19...
	 */
	public void setState (long stateA, long stateB, long stateC, long stateD, long stateE) {
		final int c = getStateCount();
		for (int i = 0; i < c; i += 5) {
			setSelectedState(i, stateA);
		}
		for (int i = 1; i < c; i += 5) {
			setSelectedState(i, stateB);
		}
		for (int i = 2; i < c; i += 5) {
			setSelectedState(i, stateC);
		}
		for (int i = 3; i < c; i += 5) {
			setSelectedState(i, stateD);
		}
		for (int i = 4; i < c; i += 5) {
			setSelectedState(i, stateE);
		}
	}

	/**
	 * Sets all state variables to alternating values chosen from {@code states}. If states is empty,
	 * then this does nothing, and leaves the current generator unchanged. This works for
	 * generators with any {@link #getStateCount()}, but may allocate an array if states is
	 * used as a varargs (you can pass an existing array without needing to allocate). This
	 * uses {@link #setSelectedState(int, long)} to change the states.
	 *
	 * @param states an array or varargs of long values to use as states
	 */
	public void setState (long... states) {
		final int c = getStateCount(), sl = states.length;
		for (int b = 0; b < sl; b++) {
			final long curr = states[b];
			for (int i = b; i < c; i += sl) {
				setSelectedState(i, curr);
			}
		}
	}

	/**
	 * 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}. *

* Note that you can give this values for {@code bits} that are outside its expected range of 1 to 32, * but the value used, as long as bits is positive, will effectively be {@code bits % 32}. As stated * before, a value of 0 for bits is the same as a value of 32.

* * @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) { return (int)nextLong() >>> 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 () { return (int)nextLong(); } /** * 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. *
* This method clamps bound to be at least 0; it never returns a negative int. *
* 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) { return (int)(bound * (nextLong() & 0xFFFFFFFFL) >> 32) & ~(bound >> 31); } /** * 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. *
* This method treats the outer bound as unsigned, so if a negative int is passed as * {@code bound}, it will be treated as positive and larger than {@link Integer#MAX_VALUE}. * That means this can produce results that are positive or negative, but when you * mask the result and the bound with {@code 0xFFFFFFFFL} (to treat them as unsigned), * the result will always be between {@code 0L} (inclusive) and the masked bound * (exclusive). *
* This is primarily useful as a building block for other methods in this class. * * @param bound the upper bound (exclusive); treated as unsigned * @return the next pseudorandom, uniformly distributed {@code int} * value between zero (inclusive) and {@code bound} (exclusive), treated as * unsigned, from this random number generator's sequence */ public int nextUnsignedInt (int bound) { return (int)((bound & 0xFFFFFFFFL) * (nextLong() & 0xFFFFFFFFL) >>> 32); } /** * 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) { outerBound = (int)(outerBound * (nextLong() & 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}. * *
For any case where outerBound might be valid but less than innerBound, you * can use {@link #nextSignedInt(int, int)}. If outerBound is less than innerBound * here, this simply returns innerBound. * * @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)(innerBound + (nextUnsignedInt(outerBound - innerBound) & ~((long)outerBound - (long)innerBound >> 63))); } /** * 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. * * @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 innerBound + nextUnsignedInt(outerBound - innerBound); } /** * 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. *
* The only methods that need to be implemented by this interface are * this and {@link #copy()}, though other methods can be implemented * as appropriate for generators that, for instance, natively produce * ints rather than longs. * * @return the next pseudorandom, uniformly distributed {@code long} * value from this random number generator's sequence */ public abstract long nextLong (); /** * 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. *
* The public implementation simply returns a sign check on {@link #nextLong()}, * returning true if the generated long is negative. This is typically the safest * way to implement this method; many types of generators have less statistical * quality on their lowest bit, so just returning based on the lowest bit isn't * always a good idea. * * @return the next pseudorandom, uniformly distributed * {@code boolean} value from this random number generator's * sequence */ public boolean nextBoolean () { return nextLong() < 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 public implementation uses the upper 24 bits of {@link #nextLong()}, * with an unsigned right shift and a multiply by a very small float * ({@code 5.9604645E-8f} or {@code 0x1p-24f}). It tends to be fast if * nextLong() is fast, but alternative implementations could use 24 bits of * {@link #nextInt()} (or just {@link #next(int)}, giving it {@code 24}) * if that generator doesn't efficiently generate 64-bit longs.

* * @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 () { return (nextLong() >>> 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 public implementation uses the upper 53 bits of {@link #nextLong()}, * with an unsigned right shift and a multiply by a very small double * ({@code 1.1102230246251565E-16}, or {@code 0x1p-53}). It should perform well * if nextLong() performs well, and is expected to perform less well if the * generator naturally produces 32 or fewer bits at a time.

* * @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 () { return (nextLong() >>> 11) * 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.000000000000011102230246251565404236316680908203125% of the time * if there is no bias in the generator, but it can happen. This uses similar code to {@link #nextExclusiveDouble()} * internally, and retains its quality of having approximately uniform distributions for every mantissa bit, unlike * most ways of generating random floating-point numbers. * * @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; final long bits = nextLong(); return BitConversion.longBitsToDouble(1022L - BitConversion.countTrailingZeros(bits) << 52 | bits >>> 12) + 0x1p-12 - 0x1p-12; // return BitConversion.longBitsToDouble(1022L - BitConversion.countLeadingZeros(bits) << 52 | (bits & 0xFFFFFFFFFFFFFL)) + 0x1p-12 - 0x1p-12; } /** * 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 does not return purely-equidistant floats, because there the resolution of possible * floats it can generate is higher as it approaches 0.0 . The smallest non-zero float this can return is * 5.421011E-20f (0x1p-64f in hex), and the largest non-one float this can return is 0.9999999f (0x1.fffffcp-1f in * hex). This uses nearly identical code to {@link #nextExclusiveFloat()}, but carefully adds and subtracts a small * number to force rounding at 0.0 and 1.0 . This retains the exclusive version's quality of having approximately * uniform distributions for every mantissa bit, unlike most ways of generating random floating-point numbers. * * @return a float between 0.0, inclusive, and 1.0, inclusive */ public float nextInclusiveFloat () { // return (int)(0x1000001L * (nextLong() & 0xFFFFFFFFL) >> 32) * 0x1p-24f; final long bits = nextLong(); return BitConversion.intBitsToFloat(126 - BitConversion.countLeadingZeros(bits) << 23 | ((int)bits & 0x7FFFFF)) + 0x1p-22f - 0x1p-22f; } /** * 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); } /** * Gets a random double between 0.0 and 1.0, exclusive at both ends; this method is also more uniform than * {@link #nextDouble()} if you use the bit-patterns of the returned doubles. This is a simplified version of * this algorithm by Allen Downey. This can return double * values between 2.710505431213761E-20 and 0.9999999999999999, or 0x1.0p-65 and 0x1.fffffffffffffp-1 in hex * notation. It cannot return 0 or 1. Some cases can prefer {@link #nextExclusiveDoubleEquidistant()}, which is * implemented more traditionally but may have slower performance. This method can also return doubles that * are extremely close to 0, but can't return doubles that are as close to 1, due to how floating-point numbers * work. However, nextExclusiveDoubleEquidistant() can return only a minimum value that is as distant from 0 as its * maximum value is distant from 1. *
* To compare, nextDouble() and nextExclusiveDoubleEquidistant() are less likely to produce a "1" bit for their * lowest 5 bits of mantissa/significand (the least significant bits numerically, but potentially important * for some uses), with the least significant bit produced half as often as the most significant bit in the * mantissa. As for this method, it has approximately the same likelihood of producing a "1" bit for any * position in the mantissa. *
* The implementation may have different performance characteristics than {@link #nextDouble()}, because this * doesn't perform any floating-point multiplication or division, and instead assembles bits obtained by one call to * {@link #nextLong()}. This uses {@link BitConversion#longBitsToDouble(long)} and * {@link BitConversion#countTrailingZeros(long)}, both of which typically have optimized intrinsics on HotSpot, and * this is branchless and loopless, unlike the original algorithm by Allen Downey. When compared with * {@link #nextExclusiveDoubleEquidistant()}, this method performs better on at least HotSpot JVMs. On GraalVM 17, * this is over twice as fast as nextExclusiveDoubleEquidistant(). * * @return a random uniform double between 2.710505431213761E-20 and 0.9999999999999999 (both inclusive) */ public double nextExclusiveDouble () { final long bits = nextLong(); return BitConversion.longBitsToDouble(1022L - BitConversion.countTrailingZeros(bits) << 52 | bits >>> 12); } // This could be used above, but it favors low results slightly. // return BitConversion.longBitsToDouble(1022L - BitConversion.countLeadingZeros(bits) << 52 | (bits & 0xFFFFFFFFFFFFFL)); /** * Gets a random double between 0.0 and 1.0, exclusive at both ends. This can return double * values between 1.1102230246251565E-16 and 0.9999999999999999, or 0x1.0p-53 and 0x1.fffffffffffffp-1 in hex * notation. It cannot return 0 or 1, and its minimum and maximum results are equally distant from 0 and from * 1, respectively. Many usages may prefer {@link #nextExclusiveDouble()}, which is better-distributed if you * consider the bit representation of the returned doubles, tends to perform better, and can return doubles that * much closer to 0 than this can. *
* The implementation simply uses {@link #nextLong(long)} to get a uniformly-chosen long between 1 and * (2 to the 53) - 1, both inclusive, and multiplies it by (2 to the -53). Using larger values than (2 to the * 53) would cause issues with the double math. * * @return a random uniform double between 0 and 1 (both exclusive) */ public double nextExclusiveDoubleEquidistant () { return (nextLong(0x1FFFFFFFFFFFFFL) + 1L) * 0x1p-53; } /** * Just like {@link #nextDouble(double)}, but this is exclusive on both 0.0 and {@code outerBound}. * Like {@link #nextExclusiveDouble()}, which this uses, this may have better bit-distribution of * double values, and it may also be better able to produce very small doubles when {@code outerBound} is large. * It should typically be a little faster than {@link #nextDouble(double)}. * * @param outerBound the outer exclusive bound; may be positive or negative * @return a double between 0.0, exclusive, and {@code outerBound}, exclusive */ public double nextExclusiveDouble (double outerBound) { return nextExclusiveDouble() * outerBound; } /** * Just like {@link #nextDouble(double, double)}, but this is exclusive on both {@code innerBound} and {@code outerBound}. * Like {@link #nextExclusiveDouble()}, which this uses,, this may have better bit-distribution of double values, * and it may also be better able to produce doubles close to innerBound when {@code outerBound - innerBound} is large. * It should typically be a little faster than {@link #nextDouble(double, double)}. * * @param innerBound the inner exclusive bound; may be positive or negative * @param outerBound the outer exclusive bound; may be positive or negative * @return a double between {@code innerBound}, exclusive, and {@code outerBound}, exclusive */ public double nextExclusiveDouble (double innerBound, double outerBound) { return innerBound + nextExclusiveDouble() * (outerBound - innerBound); } /** * Gets a random double that may be positive or negative, but cannot be 0, and always has a magnitude less than 1. *
* This is a modified version of this * algorithm by Allen Downey. This version can return double values between -0.9999999999999999 and * -5.421010862427522E-20, as well as between 2.710505431213761E-20 and 0.9999999999999999, or -0x1.fffffffffffffp-1 * to -0x1.0p-64 as well as between 0x1.0p-65 and 0x1.fffffffffffffp-1 in hex notation. It cannot return -1, 0 or 1. * It has much more uniform bit distribution across its mantissa/significand bits than {@link Random#nextDouble()}, * especially when the result of nextDouble() is expanded to the -1.0 to 1.0 range (such as with * {@code 2.0 * (nextDouble() - 0.5)}). Where the given example code is unable to produce a "1" bit for its lowest * bit of mantissa (the least significant bits numerically, but potentially important for some uses), this has * approximately the same likelihood of producing a "1" bit for any positions in the mantissa, and also equal odds * for the sign bit. * @return a random uniform double between -1 and 1 with a tiny hole around 0 (all exclusive) */ public double nextExclusiveSignedDouble(){ final long bits = nextLong(); return BitConversion.longBitsToDouble(1022L - BitConversion.countTrailingZeros(bits) << 52 | ((bits << 32 | bits >>> 32) & 0x800FFFFFFFFFFFFFL)); } // return BitConversion.longBitsToDouble(1022L - BitConversion.countLeadingZeros(bits) << 52 | ((bits << 63 | bits >>> 1) & 0x800FFFFFFFFFFFFFL)); /** * Gets a random float between 0.0 and 1.0, exclusive at both ends. This method is also more uniform than * {@link #nextFloat()} if you use the bit-patterns of the returned floats. This is a simplified version of * this algorithm by Allen Downey. This version can * return float values between 2.7105054E-20 to 0.99999994, or 0x1.0p-65 to 0x1.fffffep-1 in hex notation. * It cannot return 0 or 1. To compare, nextFloat() is less likely to produce a "1" bit for its * lowest 5 bits of mantissa/significand (the least significant bits numerically, but potentially important * for some uses), with the least significant bit produced half as often as the most significant bit in the * mantissa. As for this method, it has approximately the same likelihood of producing a "1" bit for any * position in the mantissa. *
* The implementation may have different performance characteristics than {@link #nextFloat()}, * because this doesn't perform any floating-point multiplication or division, and instead assembles bits * obtained by one call to {@link #nextLong()}. This uses {@link BitConversion#intBitsToFloat(int)} and * {@link BitConversion#countLeadingZeros(long)}, both of which typically have optimized intrinsics on HotSpot, * and this is branchless and loopless, unlike the original algorithm by Allen Downey. When compared with * {@link #nextExclusiveFloatEquidistant()}, this method performs better on at least HotSpot JVMs. On GraalVM 17, * this is over twice as fast as nextExclusiveFloatEquidistant(). * * @return a random uniform float between 0 and 1 (both exclusive) */ public float nextExclusiveFloat () { final long bits = nextLong(); return BitConversion.intBitsToFloat(126 - BitConversion.countLeadingZeros(bits) << 23 | ((int)bits & 0x7FFFFF)); } // return Float.intBitsToFloat(126 - Long.numberOfTrailingZeros(bits) << 23 | (int)(bits >>> 41)); // return Float.intBitsToFloat(126 - Long.numberOfLeadingZeros(bits) << 23 | ((int)bits & 0x7FFFFF)); /** * Gets a random float between 0.0 and 1.0, exclusive at both ends. This can return float * values between 5.9604645E-8 and 0.99999994, or 0x1.0p-24 and 0x1.fffffep-1 in hex notation. * It cannot return 0 or 1, and its minimum and maximum results are equally distant from 0 and from * 1, respectively. Most usages might prefer {@link #nextExclusiveFloat()}, which is * better-distributed if you consider the bit representation of the returned floats, tends to perform * better, and can return floats that much closer to 0 than this can. *
* The implementation simply uses {@link #nextInt(int)} to get a uniformly-chosen int between 1 and * (2 to the 24) - 1, both inclusive, and multiplies it by (2 to the -24). Using larger values than (2 to the * 24) would cause issues with the float math. * * @return a random uniform float between 0 and 1 (both exclusive) */ public float nextExclusiveFloatEquidistant () { return (nextInt(0xFFFFFF) + 1) * 0x1p-24f; } /** * Just like {@link #nextFloat(float)}, but this is exclusive on both 0.0 and {@code outerBound}. * Like {@link #nextExclusiveFloat()}, this may have better bit-distribution of float values, and * it may also be better able to produce very small floats when {@code outerBound} is large. * It should be a little faster than {@link #nextFloat(float)}. * * @param outerBound the outer exclusive bound; may be positive or negative * @return a float between 0.0, exclusive, and {@code outerBound}, exclusive */ public float nextExclusiveFloat (float outerBound) { return nextExclusiveFloat() * outerBound; } /** * Just like {@link #nextFloat(float, float)}, but this is exclusive on both {@code innerBound} and {@code outerBound}. * Like {@link #nextExclusiveFloat()}, this may have better bit-distribution of float values, and * it may also be better able to produce floats close to innerBound when {@code outerBound - innerBound} is large. * It should be a little faster than {@link #nextFloat(float, float)}. * * @param innerBound the inner exclusive bound; may be positive or negative * @param outerBound the outer exclusive bound; may be positive or negative * @return a float between {@code innerBound}, exclusive, and {@code outerBound}, exclusive */ public float nextExclusiveFloat (float innerBound, float outerBound) { return innerBound + nextExclusiveFloat() * (outerBound - innerBound); } /** * Gets a random float that may be positive or negative, but cannot be 0, and always has a magnitude less than 1. *
* This is a modified version of this * algorithm by Allen Downey. This version can return double values between -0.99999994 and -1.1641532E-10, as * well as between 2.7105054E-20 and 0.99999994, or -0x1.fffffep-1 to -0x1.0p-33 as well as between 0x1.0p-65 and * 0x1.fffffep-1 in hex notation. It cannot return -1, 0 or 1. It has much more uniform bit distribution across its * mantissa/significand bits than {@link Random#nextDouble()}, especially when the result of nextDouble() is * expanded to the -1.0 to 1.0 range (such as with {@code 2.0 * (nextDouble() - 0.5)}). Where the given example code * is unable to produce a "1" bit for its lowest bit of mantissa (the least significant bits numerically, but * potentially important for some uses), this has approximately the same likelihood of producing a "1" bit for any * positions in the mantissa, and also equal odds for the sign bit. * @return a random uniform double between -1 and 1 with a tiny hole around 0 (all exclusive) */ public float nextExclusiveSignedFloat(){ final long bits = nextLong(); return BitConversion.intBitsToFloat(126 - BitConversion.countLeadingZeros(bits) << 23 | ((int)bits & 0x807FFFFF)); } /** * 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 does not use a rough approximation, which is a departure from earlier * versions; instead, it uses the Ziggurat method, which produces high-quality * variables very quickly. Like earlier versions that used probit() or a * bit-counting approximation, this requests exactly one long from the * generator's sequence (using {@link #nextLong()}). This makes it different * from code like java.util.Random's nextGaussian() method, which can (rarely) * fetch a higher number of random doubles. *

* The implementation here was ported from code by Olaf Berstein, based on a * paper by Jorgen A. Doornik and some steps from a paper by George Marsaglia. * {@link Ziggurat} has more information, for the curious. * * @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 () { return Distributor.normal(nextLong()); } /** * Returns the next pseudorandom, Gaussian ("normally") distributed {@code double} * value with the specified mean and standard deviation from this random number generator's sequence. *
* This defaults to simply returning {@code mean + stddev * nextGaussian()}. * * @param mean the mean of the Gaussian distribution to be drawn from * @param stddev the standard deviation (square root of the variance) * of the Gaussian distribution to be drawn from * * @return a Gaussian distributed {@code double} with the specified mean and standard deviation */ public double nextGaussian(double mean, double stddev) { return mean + stddev * nextGaussian(); } /** * Optional; advances or rolls back the {@code EnhancedRandom}' 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()}. * *

The public implementation throws an UnsupportedOperationException. Many types of random * number generator do not have an efficient way of skipping arbitrarily through the state sequence, * and those types should not implement this method differently. * * @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) { throw new UnsupportedOperationException("skip() not supported."); } /** * Optional; moves the state to its previous value and returns the previous long that would have been produced by * {@link #nextLong()}. This can be equivalent to calling {@link #skip(long)} with -1L, but not always; many * generators can't efficiently skip long distances, but can step back by one value. *
* Generators that natively generate {@code int} results typically produce {@code long} values by generating an int * for the high 32 bits and an int for the low 32 bits. When producing the previous long, the order the high and low * bits are generated, such as by {@link #previousInt()}, should be reversed. Generators that natively produce * {@code long} values usually don't need to implement {@link #previousInt()}, but those that produce {@code int} * usually should implement it, and may optionally call previousInt() twice in this method. *
* If you know how to implement the reverse of a particular random number generator, it is recommended you do so * here, rather than rely on skip(). This isn't always easy, but should always be possible for any decent PRNG (some * historical PRNGs, such as the Middle-Square PRNG, cannot be reversed at all). If a generator cannot be reversed * because multiple initial states can transition to the same subsequent state, it is known to have statistical * problems that are not necessarily present in a generator that matches one initial state to one subsequent state. *
* The public implementation calls {@link #skip(long)} with -1L, and if skip() has not been implemented * differently, then it will throw an UnsupportedOperationException. * * @return the previous number this would have produced with {@link #nextLong()} */ public long previousLong () { return skip(-1L); } /** * Optional; moves the state to its previous value and returns the previous int that would have been produced by * {@link #nextInt()}. This can be equivalent to calling {@link #previousLong()} and casting to int, but not always; * generators that natively generate {@code int} results typically move the state once in nextInt() and twice in * nextLong(), and should move the state back once here. *
* If {@link #nextInt()} is implemented using a call to {@link #nextLong()}, the implementation in this class is * almost always sufficient and correct. If nextInt() changes state differently from nextLong(), then this should be * implemented, if feasible, and {@link #previousLong()} can be implemented using this method. * If you know how to implement the reverse of a particular random number generator, it is recommended you do so * here, rather than rely on skip(). This isn't always easy, but should always be possible for any decent PRNG (some * historical PRNGs, such as the Middle-Square PRNG, cannot be reversed at all). If a generator cannot be reversed * because multiple initial states can transition to the same subsequent state, it is known to have statistical * problems that are not necessarily present in a generator that matches one initial state to one subsequent state. *
* The public implementation calls {@link #previousLong()} and casts it to int, and if previousLong() and skip() * have not been implemented differently, then it will throw an UnsupportedOperationException. * * @return the previous number this would have produced with {@link #nextInt()} */ public int previousInt () { return (int)previousLong(); } /** * Creates a new EnhancedRandom with identical states to this one, so if the same EnhancedRandom 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 any parent class, so if you call methods that are * only implemented by a superclass (like {@link java.util.Random}) and not this one, the results may differ. * * @return a deep copy of this EnhancedRandom. */ public abstract EnhancedRandom copy (); /** * Similar to {@link #copy()}, but fills this EnhancedRandom with the state of another EnhancedRandom, usually * (but not necessarily) one of the same type. If this class has the same {@link #getStateCount()} as other's * class, then this method copies the full state of other into this object. Otherwise, if this class has a * larger state count than other's class, then all of other's state is copied into the same selections in this * object, and the rest of this object's state is filled with {@code -1L} using * {@link #setSelectedState(int, long)}. If this class has a smaller state count than other's class, then only * part of other's state is copied, and this method stops when all of this object's states have been assigned. *
* If this class has restrictions on its state, they will be respected by the public implementation of this * method as long as {@link #setSelectedState(int, long)} behaves correctly for those restrictions. Note that * this method will public to throwing an UnsupportedOperationException unless {@link #getSelectedState(int)} * is implemented by other so its state can be accessed. This may also behave badly if * {@link #setSelectedState(int, long)} isn't implemented (it may be fine for some cases where the state count * is 1, but don't count on it). If other's class doesn't implement {@link #getStateCount()}, then this method * sets the entire state of this object to -1L; if this class doesn't implement getStateCount(), then this * method does nothing. * * @param other another EnhancedRandom, typically with the same class as this one, to copy its state into this */ public void setWith (EnhancedRandom other) { final int myCount = getStateCount(), otherCount = other.getStateCount(); int i = 0; for (; i < myCount && i < otherCount; i++) { setSelectedState(i, other.getSelectedState(i)); } for (; i < myCount; i++) { setSelectedState(i, -1L); } } /** * A way of taking a double in the (0.0, 1.0) range and mapping it to a Gaussian or normal distribution, so high * inputs correspond to high outputs, and similarly for the low range. This is centered on 0.0 and its standard * deviation seems to be 1.0 (the same as {@link java.util.Random#nextGaussian()}). If this is given an input of 0.0 * or less, it returns -38.5, which is slightly less than the result when given {@link Double#MIN_VALUE}. If it is * given an input of 1.0 or more, it returns 38.5, which is significantly larger than the result when given the * largest double less than 1.0 (this value is further from 1.0 than {@link Double#MIN_VALUE} is from 0.0). If * given {@link Double#NaN}, it returns whatever {@link Math#copySign(double, double)} returns for the arguments * {@code 38.5, Double.NaN}, which is implementation-dependent. It uses an algorithm by Peter John Acklam, as * implemented by Sherali Karimov. * Original source. * Information on the algorithm. * Wikipedia's page on the probit function may help, but * is more likely to just be confusing. *
* Acklam's algorithm and Karimov's implementation are both quite fast. This appears faster than generating * Gaussian-distributed numbers using either the Box-Muller Transform or Marsaglia's Polar Method, though it isn't * as precise and can't produce as extreme min and max results in the extreme cases they should appear. If given * a typical uniform random {@code double} that's exclusive on 1.0, it won't produce a result higher than * {@code 8.209536145151493}, and will only produce results of at least {@code -8.209536145151493} if 0.0 is * excluded from the inputs (if 0.0 is an input, the result is {@code -38.5}). A chief advantage of using this with * a random number generator is that it only requires one random double to obtain one Gaussian value; * {@link java.util.Random#nextGaussian()} generates at least two random doubles for each two Gaussian values, but * may rarely require much more random generation. Note that this method isn't used by default for * {@link #nextGaussian()}, because it uses a very different approximation that is faster but less precise. *
* This can be used both as an optimization for generating Gaussian random values, and as a way of generating * Gaussian values that match a pattern present in the inputs (which you could have by using a sub-random sequence * as the input, such as those produced by a van der Corput, Halton, Sobol or R2 sequence). Most methods of generating * Gaussian values (e.g. Box-Muller and Marsaglia polar) do not have any way to preserve a particular pattern. * * @param d should be between 0 and 1, exclusive, but other values are tolerated * @return a normal-distributed double centered on 0.0; all results will be between -38.5 and 38.5, both inclusive */ public static double probit (final double d) { if (d <= 0 || d >= 1) { return Math.copySign(38.5, d - 0.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); } 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); } /** * Attempts to improve the quality of a "gamma" increment for an additive sequence. This is stricter than the checks * in Java 8's SplittableRandom. The goal here is to make sure the gamma is "sufficiently random" to avoid patterns * when used as an increment. Examples of gamma values that aren't random enough include {@code 1L}, {@code 3L}, * {@code 0xFFFFFFFFFFFFFFFFL}, {@code 0xAAAAAAAAAAAAAAABL}, and so on. It rejects any gamma value where any of four * bit counts are less than 24 or greater than 40. The values that have their bits counted are: *

    *
  • The gamma itself,
  • *
  • The Gray code of the gamma, defined as {@code (gamma ^ (gamma >>> 1))},
  • *
  • The {@link MathTools#modularMultiplicativeInverse(long)} of the gamma,
  • *
  • And the Gray code of the above inverse of the gamma.
  • *
* If a gamma is rejected, this multiplies it by an LCG constant, 0xD1342543DE82EF95L, adds an increasing even * number (first 2, then 4, then 6, and so on) and tries again repeatedly. It returns the first gamma that wasn't * rejected, which could be the original gamma. * * @see This was informed by O'Neill's blog post about SplittableRandom's gamma. * @param gamma any long, though almost always an odd number, that would be added as an increment in a sequence * @return gamma or a modification upon it such that its bits are "sufficiently random" to be a good increment */ public static long fixGamma(long gamma) { long inverse = MathTools.modularMultiplicativeInverse(gamma |= 1L), add = 0L; while (Math.abs(Long.bitCount(gamma) - 32) > 8 || Math.abs(Long.bitCount(gamma ^ gamma >>> 1) - 32) > 8 || Math.abs(Long.bitCount(inverse) - 32) > 8 || Math.abs(Long.bitCount(inverse ^ inverse >>> 1) - 32) > 8) { inverse = MathTools.modularMultiplicativeInverse(gamma = gamma * 0xD1342543DE82EF95L + (add += 2L)); } return gamma; } /** * Given two EnhancedRandom objects that could have the same or different classes, * this returns true if they have the same class and same state, or false otherwise. * Both of the arguments should implement {@link #getSelectedState(int)}, or this * will throw an UnsupportedOperationException. This can be useful for comparing * EnhancedRandom classes that do not implement equals(), for whatever reason. * This returns true if both arguments are null, but false if only one is null. * * @param left an EnhancedRandom to compare for equality * @param right another EnhancedRandom to compare for equality * @return true if the two EnhancedRandom objects have the same class and state, or false otherwise */ public static boolean areEqual (EnhancedRandom left, EnhancedRandom right) { if (left == right) return true; if(left == null || right == null) return false; if (left.getClass() != right.getClass()) return false; final int count = left.getStateCount(); for (int i = 0; i < count; i++) { if (left.getSelectedState(i) != right.getSelectedState(i)) return false; } return true; } /** * 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) nextTriangular(-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) nextTriangular(-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 to {@link #nextTriangular(float, float, float) nextTriangular(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)); } /** * Returns the minimum result of {@code trials} calls to {@link #nextSignedInt(int, int)} using the given {@code innerBound} * and {@code outerBound}. The innerBound is inclusive; the outerBound is exclusive. * The higher trials is, the lower the average value this returns. * * @param innerBound the inner inclusive bound; may be positive or negative * @param outerBound the outer exclusive bound; may be positive or negative * @param trials how many random numbers to acquire and compare * @return the lowest random number between innerBound (inclusive) and outerBound (exclusive) this found */ public int minIntOf (int innerBound, int outerBound, int trials) { int v = nextSignedInt(innerBound, outerBound); for (int i = 1; i < trials; i++) { v = Math.min(v, nextSignedInt(innerBound, outerBound)); } return v; } /** * Returns the maximum result of {@code trials} calls to {@link #nextSignedInt(int, int)} using the given {@code innerBound} * and {@code outerBound}. The innerBound is inclusive; the outerBound is exclusive. * The higher trials is, the higher the average value this returns. * * @param innerBound the inner inclusive bound; may be positive or negative * @param outerBound the outer exclusive bound; may be positive or negative * @param trials how many random numbers to acquire and compare * @return the highest random number between innerBound (inclusive) and outerBound (exclusive) this found */ public int maxIntOf (int innerBound, int outerBound, int trials) { int v = nextSignedInt(innerBound, outerBound); for (int i = 1; i < trials; i++) { v = Math.max(v, nextSignedInt(innerBound, outerBound)); } return v; } /** * Returns the minimum result of {@code trials} calls to {@link #nextSignedLong(long, long)} using the given {@code innerBound} * and {@code outerBound}. The innerBound is inclusive; the outerBound is exclusive. * The higher trials is, the lower the average value this returns. * * @param innerBound the inner inclusive bound; may be positive or negative * @param outerBound the outer exclusive bound; may be positive or negative * @param trials how many random numbers to acquire and compare * @return the lowest random number between innerBound (inclusive) and outerBound (exclusive) this found */ public long minLongOf (long innerBound, long outerBound, int trials) { long v = nextSignedLong(innerBound, outerBound); for (int i = 1; i < trials; i++) { v = Math.min(v, nextSignedLong(innerBound, outerBound)); } return v; } /** * Returns the maximum result of {@code trials} calls to {@link #nextSignedLong(long, long)} using the given {@code innerBound} * and {@code outerBound}. The innerBound is inclusive; the outerBound is exclusive. * The higher trials is, the higher the average value this returns. * * @param innerBound the inner inclusive bound; may be positive or negative * @param outerBound the outer exclusive bound; may be positive or negative * @param trials how many random numbers to acquire and compare * @return the highest random number between innerBound (inclusive) and outerBound (exclusive) this found */ public long maxLongOf (long innerBound, long outerBound, int trials) { long v = nextSignedLong(innerBound, outerBound); for (int i = 1; i < trials; i++) { v = Math.max(v, nextSignedLong(innerBound, outerBound)); } return v; } /** * Returns the minimum result of {@code trials} calls to {@link #nextDouble(double, double)} using the given {@code innerBound} * and {@code outerBound}. The innerBound is inclusive; the outerBound is exclusive. * The higher trials is, the lower the average value this returns. * * @param innerBound the inner inclusive bound; may be positive or negative * @param outerBound the outer exclusive bound; may be positive or negative * @param trials how many random numbers to acquire and compare * @return the lowest random number between innerBound (inclusive) and outerBound (exclusive) this found */ public double minDoubleOf (double innerBound, double outerBound, int trials) { double v = nextDouble(innerBound, outerBound); for (int i = 1; i < trials; i++) { v = Math.min(v, nextDouble(innerBound, outerBound)); } return v; } /** * Returns the maximum result of {@code trials} calls to {@link #nextDouble(double, double)} using the given {@code innerBound} * and {@code outerBound}. The innerBound is inclusive; the outerBound is exclusive. * The higher trials is, the higher the average value this returns. * * @param innerBound the inner inclusive bound; may be positive or negative * @param outerBound the outer exclusive bound; may be positive or negative * @param trials how many random numbers to acquire and compare * @return the highest random number between innerBound (inclusive) and outerBound (exclusive) this found */ public double maxDoubleOf (double innerBound, double outerBound, int trials) { double v = nextDouble(innerBound, outerBound); for (int i = 1; i < trials; i++) { v = Math.max(v, nextDouble(innerBound, outerBound)); } return v; } /** * Returns the minimum result of {@code trials} calls to {@link #nextFloat(float, float)} using the given {@code innerBound} * and {@code outerBound}. The innerBound is inclusive; the outerBound is exclusive. * The higher trials is, the lower the average value this returns. * * @param innerBound the inner inclusive bound; may be positive or negative * @param outerBound the outer exclusive bound; may be positive or negative * @param trials how many random numbers to acquire and compare * @return the lowest random number between innerBound (inclusive) and outerBound (exclusive) this found */ public float minFloatOf (float innerBound, float outerBound, int trials) { float v = nextFloat(innerBound, outerBound); for (int i = 1; i < trials; i++) { v = Math.min(v, nextFloat(innerBound, outerBound)); } return v; } /** * Returns the maximum result of {@code trials} calls to {@link #nextFloat(float, float)} using the given {@code innerBound} * and {@code outerBound}. The innerBound is inclusive; the outerBound is exclusive. * The higher trials is, the higher the average value this returns. * * @param innerBound the inner inclusive bound; may be positive or negative * @param outerBound the outer exclusive bound; may be positive or negative * @param trials how many random numbers to acquire and compare * @return the highest random number between innerBound (inclusive) and outerBound (exclusive) this found */ public float maxFloatOf (float innerBound, float outerBound, int trials) { float v = nextFloat(innerBound, outerBound); for (int i = 1; i < trials; i++) { v = Math.max(v, nextFloat(innerBound, outerBound)); } return v; } /** * 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)]; } /** * Gets a randomly selected item from the given List, such as an ArrayList. * If the List is empty, this throws an IndexOutOfBoundsException. * * @param list a non-empty implementation of List, such as ArrayList * @param the type of items * @return a randomly-selected item from list */ public T randomElement (List list) { return list.get(nextInt(list.size())); } /** * 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 a section of the given array in-place pseudo-randomly, using this to determine how to shuffle. * * @param items an int array; must be non-null * @param offset the index of the first element of the array that can be shuffled * @param length the length of the section to shuffle */ public void shuffle (int[] items, int offset, int length) { offset = Math.min(Math.max(0, offset), items.length); length = Math.min(items.length - offset, Math.max(0, length)); for (int i = offset + length - 1; i > offset; i--) { int ii = offset + nextInt(i + 1 - offset); 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 a section of the given array in-place pseudo-randomly, using this to determine how to shuffle. * * @param items a long array; must be non-null * @param offset the index of the first element of the array that can be shuffled * @param length the length of the section to shuffle */ public void shuffle (long[] items, int offset, int length) { offset = Math.min(Math.max(0, offset), items.length); length = Math.min(items.length - offset, Math.max(0, length)); for (int i = offset + length - 1; i > offset; i--) { int ii = offset + nextInt(i + 1 - offset); 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 a section of the given array in-place pseudo-randomly, using this to determine how to shuffle. * * @param items a float array; must be non-null * @param offset the index of the first element of the array that can be shuffled * @param length the length of the section to shuffle */ public void shuffle (float[] items, int offset, int length) { offset = Math.min(Math.max(0, offset), items.length); length = Math.min(items.length - offset, Math.max(0, length)); for (int i = offset + length - 1; i > offset; i--) { int ii = offset + nextInt(i + 1 - offset); 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 a section of the given array in-place pseudo-randomly, using this to determine how to shuffle. * * @param items a char array; must be non-null * @param offset the index of the first element of the array that can be shuffled * @param length the length of the section to shuffle */ public void shuffle (char[] items, int offset, int length) { offset = Math.min(Math.max(0, offset), items.length); length = Math.min(items.length - offset, Math.max(0, length)); for (int i = offset + length - 1; i > offset; i--) { int ii = offset + nextInt(i + 1 - offset); 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 a section of the given array in-place pseudo-randomly, using this to determine how to shuffle. * * @param items a byte array; must be non-null * @param offset the index of the first element of the array that can be shuffled * @param length the length of the section to shuffle */ public void shuffle (byte[] items, int offset, int length) { offset = Math.min(Math.max(0, offset), items.length); length = Math.min(items.length - offset, Math.max(0, length)); for (int i = offset + length - 1; i > offset; i--) { int ii = offset + nextInt(i + 1 - offset); 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 a section of the given array in-place pseudo-randomly, using this to determine how to shuffle. * * @param items a double array; must be non-null * @param offset the index of the first element of the array that can be shuffled * @param length the length of the section to shuffle */ public void shuffle (double[] items, int offset, int length) { offset = Math.min(Math.max(0, offset), items.length); length = Math.min(items.length - offset, Math.max(0, length)); for (int i = offset + length - 1; i > offset; i--) { int ii = offset + nextInt(i + 1 - offset); 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 a section of the given array in-place pseudo-randomly, using this to determine how to shuffle. * * @param items a short array; must be non-null * @param offset the index of the first element of the array that can be shuffled * @param length the length of the section to shuffle */ public void shuffle (short[] items, int offset, int length) { offset = Math.min(Math.max(0, offset), items.length); length = Math.min(items.length - offset, Math.max(0, length)); for (int i = offset + length - 1; i > offset; i--) { int ii = offset + nextInt(i + 1 - offset); 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 a section of the given array in-place pseudo-randomly, using this to determine how to shuffle. * * @param items a boolean array; must be non-null * @param offset the index of the first element of the array that can be shuffled * @param length the length of the section to shuffle */ public void shuffle (boolean[] items, int offset, int length) { offset = Math.min(Math.max(0, offset), items.length); length = Math.min(items.length - offset, Math.max(0, length)); for (int i = offset + length - 1; i > offset; i--) { int ii = offset + nextInt(i + 1 - offset); 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; } } /** * Shuffles a section of 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 * @param offset the index of the first element of the array that can be shuffled * @param length the length of the section to shuffle */ public void shuffle (T[] items, int offset, int length) { offset = Math.min(Math.max(0, offset), items.length); length = Math.min(items.length - offset, Math.max(0, length)); for (int i = offset + length - 1; i > offset; i--) { int ii = offset + nextInt(i + 1 - offset); T temp = items[i]; items[i] = items[ii]; items[ii] = temp; } } /** * Shuffles the given List in-place pseudo-randomly, using this to determine how to shuffle. * * @param items a List of some type {@code T}; must be non-null but may contain null items */ public void shuffle (List items) { for (int i = items.size() - 1; i > 0; i--) { int ii = nextInt(i + 1); T temp = items.get(i); items.set(i, items.get(ii)); items.set(ii, temp); } } /** * Shuffles a section of the given List in-place pseudo-randomly, using this to determine how to shuffle. * * @param items a List of some type {@code T}; must be non-null but may contain null items * @param offset the index of the first element of the array that can be shuffled * @param length the length of the section to shuffle */ public void shuffle (List items, int offset, int length) { offset = Math.min(Math.max(0, offset), items.size()); length = Math.min(items.size() - offset, Math.max(0, length)); for (int i = offset + length - 1; i > offset; i--) { int ii = offset + nextInt(i + 1 - offset); T temp = items.get(i); items.set(i, items.get(ii)); items.set(ii, temp); } } /** * Serializes the current state of this EnhancedRandom to a String that can be used by * {@link #stringDeserialize(String)} to load this state at another time. This always uses * {@link Base#BASE16} for its conversions. * @return a String storing all data from the EnhancedRandom part of this generator */ public String stringSerialize() { return stringSerialize(Base.BASE16); } /** * Serializes the current state of this EnhancedRandom to a String that can be used by * {@link #stringDeserialize(String)} to load this state at another time. * @param base which Base to use, from the "digital" library, such as {@link Base#BASE10} * @return a String storing all data from the EnhancedRandom part of this generator */ public String stringSerialize(Base base) { StringBuilder ser = new StringBuilder(getTag()); ser.append('`'); if (getStateCount() > 0) { for (int i = 0; i < getStateCount() - 1; i++) { base.appendSigned(ser, getSelectedState(i)).append('~'); } base.appendSigned(ser, getSelectedState(getStateCount() - 1)); } ser.append('`'); return ser.toString(); } /** * Given a String in the format produced by {@link #stringSerialize()}, this will attempt to set this EnhancedRandom * object to match the state in the serialized data. This only works if this EnhancedRandom is the same * implementation that was serialized. Always uses {@link Base#BASE16}. Returns this EnhancedRandom, after possibly * changing its state. * @param data a String probably produced by {@link #stringSerialize()} * @return this, after setting its state */ public EnhancedRandom stringDeserialize(String data) { return stringDeserialize(data, Base.BASE16); } /** * Given a String in the format produced by {@link #stringSerialize(Base)}, and the same {@link Base} used by * the serialization, this will attempt to set this EnhancedRandom object to match the state in the serialized * data. This only works if this EnhancedRandom is the same implementation that was serialized, and also needs * the Bases to be identical. Returns this EnhancedRandom, after possibly changing its state. * @param data a String probably produced by {@link #stringSerialize(Base)} * @param base which Base to use, from the "digital" library, such as {@link Base#BASE10} * @return this, after setting its state */ public EnhancedRandom stringDeserialize(String data, Base base) { if (getStateCount() > 0) { int idx = data.indexOf('`'); for (int i = 0; i < getStateCount() - 1; i++) setSelectedState(i, base.readLong(data, idx + 1, (idx = data.indexOf('~', idx + 1)))); setSelectedState(getStateCount() - 1, base.readLong(data, idx + 1, data.indexOf('`', idx + 1))); } return this; } /** * The object implements the writeExternal method to save its contents * by calling the methods of DataOutput for its primitive values or * calling the writeObject method of ObjectOutput for objects, strings, * and arrays. * * @param out the stream to write the object to * @throws IOException Includes any I/O exceptions that may occur * @serialData

    *
  • int stateCount; the number of states this EnhancedRandom has
  • *
  • Repeat {@code stateCount} times: *
      *
    • long state_n; the nth state used here.
    • *
    *
  • *
*/ @GwtIncompatible public void writeExternal(ObjectOutput out) throws IOException { final int states = getStateCount(); out.writeInt(states); for (int i = 0; i < states; i++) { out.writeLong(getSelectedState(i)); } } /** * The object implements the readExternal method to restore its * contents by calling the methods of DataInput for primitive * types and readObject for objects, strings and arrays. The * readExternal method must read the values in the same sequence * and with the same types as were written by writeExternal. * * @param in the stream to read data from in order to restore the object * @throws IOException if I/O errors occur */ @GwtIncompatible public void readExternal(ObjectInput in) throws IOException, ClassNotFoundException { final int states = in.readInt(); for (int i = 0; i < states; i++) { setSelectedState(i, in.readLong()); } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy