Added XorshiftRandom
This commit is contained in:
		
							parent
							
								
									f428f41cca
								
							
						
					
					
						commit
						a89bab3863
					
				| 
						 | 
				
			
			@ -32,7 +32,7 @@ import com.github.f4b6a3.ulid.guid.GuidCreator;
 | 
			
		|||
/**
 | 
			
		||||
 * A factory for Universally Unique Lexicographically Sortable Identifiers.
 | 
			
		||||
 * 
 | 
			
		||||
 * @see The ULID spec; https://github.com/ulid/spec
 | 
			
		||||
 * See the ULID spec: https://github.com/ulid/spec
 | 
			
		||||
 */
 | 
			
		||||
public class UlidCreator {
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -0,0 +1,82 @@
 | 
			
		|||
/*
 | 
			
		||||
 * MIT License
 | 
			
		||||
 * 
 | 
			
		||||
 * Copyright (c) 2018-2020 Fabio Lima
 | 
			
		||||
 * 
 | 
			
		||||
 * Permission is hereby granted, free of charge, to any person obtaining a copy
 | 
			
		||||
 * of this software and associated documentation files (the "Software"), to deal
 | 
			
		||||
 * in the Software without restriction, including without limitation the rights
 | 
			
		||||
 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
 | 
			
		||||
 * copies of the Software, and to permit persons to whom the Software is
 | 
			
		||||
 * furnished to do so, subject to the following conditions:
 | 
			
		||||
 * 
 | 
			
		||||
 * The above copyright notice and this permission notice shall be included in all
 | 
			
		||||
 * copies or substantial portions of the Software.
 | 
			
		||||
 * 
 | 
			
		||||
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 | 
			
		||||
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 | 
			
		||||
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
 | 
			
		||||
 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 | 
			
		||||
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
 | 
			
		||||
 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
 | 
			
		||||
 * SOFTWARE.
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
package com.github.f4b6a3.ulid.random;
 | 
			
		||||
 | 
			
		||||
import java.util.Random;
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * A subclass of {@link java.util.Random} that implements the Xorshift random
 | 
			
		||||
 * number generator.
 | 
			
		||||
 * 
 | 
			
		||||
 * https://en.wikipedia.org/wiki/Xorshift
 | 
			
		||||
 * 
 | 
			
		||||
 * Reference:
 | 
			
		||||
 * 
 | 
			
		||||
 * George Marsaglia. 2003. Xorshift RNGs. Journal of Statistical Software 8, 14
 | 
			
		||||
 * (2003), 1–6. https://www.jstatsoft.org/article/view/v008i14
 | 
			
		||||
 * 
 | 
			
		||||
 */
 | 
			
		||||
public class XorshiftRandom extends Random {
 | 
			
		||||
 | 
			
		||||
	private static final long serialVersionUID = 5084310156945573858L;
 | 
			
		||||
 | 
			
		||||
	private long seed;
 | 
			
		||||
	private static int count;
 | 
			
		||||
 | 
			
		||||
	public XorshiftRandom() {
 | 
			
		||||
		this((int) System.nanoTime());
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	/**
 | 
			
		||||
	 * Constructor that receives an integer as 'salt'. This value is combined
 | 
			
		||||
	 * with the current milliseconds to generate the seed.
 | 
			
		||||
	 * 
 | 
			
		||||
	 * @param salt
 | 
			
		||||
	 *            a number used to generate the seed.
 | 
			
		||||
	 */
 | 
			
		||||
	public XorshiftRandom(int salt) {
 | 
			
		||||
		long time = System.currentTimeMillis() + count++;
 | 
			
		||||
		this.seed = (((long) salt) << 32) | (time & 0x00000000ffffffffL);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	public XorshiftRandom(long seed) {
 | 
			
		||||
		this.seed = seed;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	@Override
 | 
			
		||||
	protected int next(int bits) {
 | 
			
		||||
		return (int) (nextLong() >>> (64 - bits));
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	@Override
 | 
			
		||||
	public long nextLong() {
 | 
			
		||||
		long x = this.seed;
 | 
			
		||||
		x ^= (x << 13);
 | 
			
		||||
		x ^= (x >>> 7);
 | 
			
		||||
		x ^= (x << 17);
 | 
			
		||||
		this.seed = x;
 | 
			
		||||
		return x;
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
| 
						 | 
				
			
			@ -209,8 +209,8 @@ public class Base32Util {
 | 
			
		|||
	/**
 | 
			
		||||
	 * Convert a string to an array of bytes using UTF-8.
 | 
			
		||||
	 * 
 | 
			
		||||
	 * @param string
 | 
			
		||||
	 * @return
 | 
			
		||||
	 * @param string a string
 | 
			
		||||
	 * @return a string
 | 
			
		||||
	 */
 | 
			
		||||
	public static byte[] toBytes(String string) {
 | 
			
		||||
		return string.getBytes(StandardCharsets.UTF_8);
 | 
			
		||||
| 
						 | 
				
			
			@ -219,8 +219,8 @@ public class Base32Util {
 | 
			
		|||
	/**
 | 
			
		||||
	 * Convert an array of bytes to a string using UTF-8.
 | 
			
		||||
	 * 
 | 
			
		||||
	 * @param bytes
 | 
			
		||||
	 * @return
 | 
			
		||||
	 * @param bytes a byte sequence
 | 
			
		||||
	 * @return a string
 | 
			
		||||
	 */
 | 
			
		||||
	public static String toString(byte[] bytes) {
 | 
			
		||||
		return new String(bytes, StandardCharsets.UTF_8);
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -143,11 +143,10 @@ public class UlidUtil {
 | 
			
		|||
	 * 
 | 
			
		||||
	 * The validation mode is not strict.
 | 
			
		||||
	 * 
 | 
			
		||||
	 * @see {@link UlidUtil#validate(String, boolean)}.
 | 
			
		||||
	 * See {@link UlidUtil#validate(String, boolean)}.
 | 
			
		||||
	 * 
 | 
			
		||||
	 * @param ulid
 | 
			
		||||
	 *            a ULID
 | 
			
		||||
	 * @return boolean true if valid
 | 
			
		||||
	 */
 | 
			
		||||
	protected static void validate(String ulid) {
 | 
			
		||||
		validate(ulid, false);
 | 
			
		||||
| 
						 | 
				
			
			@ -156,11 +155,10 @@ public class UlidUtil {
 | 
			
		|||
	/**
 | 
			
		||||
	 * Checks if the ULID string is a valid.
 | 
			
		||||
	 * 
 | 
			
		||||
	 * @see {@link UlidUtil#validate(String, boolean)}.
 | 
			
		||||
	 * See {@link UlidUtil#validate(String, boolean)}.
 | 
			
		||||
	 * 
 | 
			
		||||
	 * @param ulid
 | 
			
		||||
	 *            a ULID
 | 
			
		||||
	 * @return boolean true if valid
 | 
			
		||||
	 */
 | 
			
		||||
	protected static void validate(String ulid, boolean strict) {
 | 
			
		||||
		if (!isValid(ulid, strict)) {
 | 
			
		||||
| 
						 | 
				
			
			@ -173,7 +171,7 @@ public class UlidUtil {
 | 
			
		|||
	 * 
 | 
			
		||||
	 * The validation mode is not strict.
 | 
			
		||||
	 * 
 | 
			
		||||
	 * @see {@link UlidUtil#validate(String, boolean)}.
 | 
			
		||||
	 * See {@link UlidUtil#validate(String, boolean)}.
 | 
			
		||||
	 */
 | 
			
		||||
	public static boolean isValid(String ulid) {
 | 
			
		||||
		return isValid(ulid, false);
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -9,6 +9,23 @@ public class NaiveRandomTest {
 | 
			
		|||
	private static final int DEFAULT_LOOP_LIMIT = 100_000;
 | 
			
		||||
	private static final String EXPECTED_BIT_COUNT_RANDOM_LONG = "The average bit count expected for random long values is 32";
 | 
			
		||||
 | 
			
		||||
	@Test
 | 
			
		||||
	public void testXorshiftNextLongNaiveAverageBitCount() {
 | 
			
		||||
		
 | 
			
		||||
		double accumulator = 0;
 | 
			
		||||
		
 | 
			
		||||
		XorshiftRandom random = new XorshiftRandom();
 | 
			
		||||
		
 | 
			
		||||
		for(int i = 0; i < DEFAULT_LOOP_LIMIT; i++) {
 | 
			
		||||
			long value = random.nextLong();
 | 
			
		||||
			accumulator += Long.bitCount(value);
 | 
			
		||||
		}
 | 
			
		||||
		
 | 
			
		||||
		double average = Math.round(accumulator / DEFAULT_LOOP_LIMIT);
 | 
			
		||||
		
 | 
			
		||||
		assertTrue(EXPECTED_BIT_COUNT_RANDOM_LONG, average == 32);
 | 
			
		||||
	}
 | 
			
		||||
	
 | 
			
		||||
	@Test
 | 
			
		||||
	public void testXorshift128PlusNextLongNaiveAverageBitCount() {
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
		Loading…
	
		Reference in New Issue