com.github.tommyettinger.random.KnownSequenceRandom Maven / Gradle / Ivy
Show all versions of juniper Show documentation
package com.github.tommyettinger.random;
import com.github.tommyettinger.digital.Base;
import java.util.Random;
/**
* A non-random number generator that simply repeats the next of a sequence of {@code long} values
* every time {@link #nextLong()} is called. Because other methods rely on nextLong()'s exact output,
* even if they don't use all of it, storing long values is usually enough to repeat any sequence of
* calls made to a generator. This is meant to be useful for things like unit tests or procedural
* generation, where a particular group of outputs should be exactly replicable.
*/
public class KnownSequenceRandom extends EnhancedRandom {
public LongSequence known;
public int index;
public KnownSequenceRandom() {
super(123456789L);
known = new LongSequence();
index = 0;
}
public KnownSequenceRandom(LongSequence seq) {
super(123456789L);
known = seq;
index = 0;
}
public KnownSequenceRandom(LongSequence seq, int position) {
super(123456789L);
known = seq;
index = (int) (position & 0x7FFFFFFFL) % known.size;
}
/**
* Gets the tag used to identify this type of EnhancedRandom, which is "KnSR".
*
* @return the constant String "KnSR"
*/
@Override
public String getTag() {
return "KnSR";
}
/**
* Sets the position of the iteration this makes through its known sequence.
*
* @param position usually a positive int less than {@code known.size}, but this technically can be any long
*/
@Override
public void setSeed(long position) {
if(known == null) return;
index = (int) (position & 0x7FFFFFFFL) % known.size;
}
/**
* Returns the next {@code long} value from this generator's sequence.
* This "generator" only cycles through a known sequence of values; it
* does not actually do any math to generate random numbers.
*
* All other "random" number generation methods in this class call this
* method, so if another class also relies on nextLong() for all
* randomness, then recording each of those nextLong() outputs in a
* LongSequence will allow a section of a generator to be played back
* more or less exactly.
*
* @return the next long from the known sequence
*/
@Override
public long nextLong() {
final long r = known.get(index++);
if(index == known.size) index = 0;
return r;
}
/**
* 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 Random}) and not this one, the results may differ.
*
* @return a deep copy of this EnhancedRandom.
*/
@Override
public KnownSequenceRandom copy() {
return new KnownSequenceRandom(known.copy(), index);
}
/**
* Returns 1, referring to the one state this changes on its own ({@link #index}).
* This does not include the potentially many values in the known sequence.
*
* @return one (1)
*/
@Override
public int getStateCount() {
return 1;
}
/**
* Gets the current index/position in the known sequence.
*
* @param selection ignored
* @return the exact value of {@link #index}
*/
@Override
public long getSelectedState(int selection) {
return index;
}
/**
* Sets the index/position in the known sequence, if {@code value} is at least equal to 0 and less than
* {@code known.size}. If value is outside that range, this can assign any value inside the range to
* the index. If {@code known.size} is 0 or less, this always assigns 0 to index (anticipating some change
* to the known sequence before it is used, hopefully).
*
* @param selection ignored
* @param value the value to use for index, if at least equal to 0 and less than {@code known.size}
*/
@Override
public void setSelectedState(int selection, long value) {
if(known.size <= 0) index = 0;
else index = (int) (value & 0x7FFFFFFFL) % known.size;
}
/**
* Sets the index/position in the known sequence, if {@code state} is at least equal to 0 and less than
* {@code known.size}. If state is outside that range, this can assign any value inside the range to
* the index. If {@code known.size} is 0 or less, this always assigns 0 to index (anticipating some change
* to the known sequence before it is used, hopefully).
*
* @param state the value to use for index, if at least equal to 0 and less than {@code known.size}
*/
@Override
public void setState(long state) {
if(known.size <= 0) index = 0;
else index = (int) (state & 0x7FFFFFFFL) % known.size;
}
/**
* Optional; moves the state to its previous value and returns the previous long that would have been produced by
* {@link #nextLong()}. This is often equivalent to calling {@link #skip(long)} with -1L, but not always; some
* generators can't efficiently skip long distances, but can step back by one value.
*
* 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()}
*/
@Override
public long previousLong() {
final long r = known.get(index--);
if(index == -1) index = known.size - 1;
return r;
}
public StringBuilder appendSerialized(StringBuilder sb, Base base) {
sb.append(getTag()).append('`');
base.appendSigned(sb, index);
sb.append('~');
known.appendSerialized(sb, base);
return sb.append('`');
}
public StringBuilder appendSerialized(StringBuilder sb) {
return appendSerialized(sb, Base.BASE10);
}
/**
* 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
*/
@Override
public String stringSerialize(Base base) {
return appendSerialized(new StringBuilder(), base).toString();
}
/**
* 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
*/
@Override
public KnownSequenceRandom stringDeserialize(String data, Base base) {
int tilde;
index = base.readInt(data, data.indexOf('`')+1, tilde = data.indexOf('~'));
known.stringDeserialize(data.substring(tilde+1), base);
return this;
}
@Override
public String toString() {
return "KnownSequenceRandom{" +
"at index=" + index +
" in known=" + known +
'}';
}
@Override
public boolean equals(Object o) {
if (this == o) return true;
if (o == null || getClass() != o.getClass()) return false;
KnownSequenceRandom that = (KnownSequenceRandom) o;
if (index != that.index) return false;
return known.equals(that.known);
}
@Override
public int hashCode() {
int result = known.hashCode();
result = 31 * result + index;
return result;
}
}