From 0412ad3df0b13ee08f4ef561d50137b93a2b1332 Mon Sep 17 00:00:00 2001 From: samy Date: Mon, 31 May 2021 20:37:50 +0200 Subject: [PATCH] MATH-1597: LowDiscrepancySequence supplier/jump for Halton and Sobol --- .../HaltonSequenceGenerator.java | 61 +++++++++++++------ .../quasirandom/LowDiscrepancySequence.java | 32 ++++++++++ .../SobolSequenceGenerator.java | 56 +++++++++++------ .../legacy/quasirandom/package-info.java | 17 ++++++ .../HaltonSequenceGeneratorTest.java | 31 +++++++--- .../SobolSequenceGeneratorTest.java | 32 ++++++---- 6 files changed, 171 insertions(+), 58 deletions(-) rename commons-math-legacy/src/main/java/org/apache/commons/math4/legacy/{random => quasirandom}/HaltonSequenceGenerator.java (80%) create mode 100644 commons-math-legacy/src/main/java/org/apache/commons/math4/legacy/quasirandom/LowDiscrepancySequence.java rename commons-math-legacy/src/main/java/org/apache/commons/math4/legacy/{random => quasirandom}/SobolSequenceGenerator.java (88%) create mode 100644 commons-math-legacy/src/main/java/org/apache/commons/math4/legacy/quasirandom/package-info.java rename commons-math-legacy/src/test/java/org/apache/commons/math4/legacy/{random => quasirandom}/HaltonSequenceGeneratorTest.java (85%) rename commons-math-legacy/src/test/java/org/apache/commons/math4/legacy/{random => quasirandom}/SobolSequenceGeneratorTest.java (82%) diff --git a/commons-math-legacy/src/main/java/org/apache/commons/math4/legacy/random/HaltonSequenceGenerator.java b/commons-math-legacy/src/main/java/org/apache/commons/math4/legacy/quasirandom/HaltonSequenceGenerator.java similarity index 80% rename from commons-math-legacy/src/main/java/org/apache/commons/math4/legacy/random/HaltonSequenceGenerator.java rename to commons-math-legacy/src/main/java/org/apache/commons/math4/legacy/quasirandom/HaltonSequenceGenerator.java index 8a175618bf..e698ea4a3b 100644 --- a/commons-math-legacy/src/main/java/org/apache/commons/math4/legacy/random/HaltonSequenceGenerator.java +++ b/commons-math-legacy/src/main/java/org/apache/commons/math4/legacy/quasirandom/HaltonSequenceGenerator.java @@ -14,14 +14,15 @@ * See the License for the specific language governing permissions and * limitations under the License. */ -package org.apache.commons.math4.legacy.random; - -import java.util.function.Supplier; +package org.apache.commons.math4.legacy.quasirandom; import org.apache.commons.math4.legacy.exception.DimensionMismatchException; +import org.apache.commons.math4.legacy.exception.NotPositiveException; import org.apache.commons.math4.legacy.exception.NullArgumentException; import org.apache.commons.math4.legacy.exception.OutOfRangeException; +import java.util.Arrays; + /** * Implementation of a Halton sequence. *

@@ -43,7 +44,7 @@ * The generator supports two modes: *

* * @see Halton sequence (Wikipedia) @@ -51,7 +52,7 @@ * On the Halton sequence and its scramblings * @since 3.3 */ -public class HaltonSequenceGenerator implements Supplier { +public class HaltonSequenceGenerator implements LowDiscrepancySequence { /** The first 40 primes. */ private static final int[] PRIMES = new int[] { @@ -71,7 +72,7 @@ public class HaltonSequenceGenerator implements Supplier { private final int dimension; /** The current index in the sequence. */ - private int count; + private long count; /** The base numbers for each component. */ private final int[] base; @@ -89,6 +90,13 @@ public HaltonSequenceGenerator(final int dimension) { this(dimension, PRIMES, WEIGHTS); } + private HaltonSequenceGenerator(HaltonSequenceGenerator original){ + this.base = original.base; + this.count = original.count; + this.dimension = original.dimension; + this.weight = Arrays.copyOf(original.weight,original.weight.length); + } + /** * Construct a new Halton sequence generator with the given base numbers and weights for each dimension. * The length of the bases array defines the space dimension and is required to be > 0. @@ -123,12 +131,12 @@ public HaltonSequenceGenerator(final int dimension, final int[] bases, final int public double[] get() { final double[] v = new double[dimension]; for (int i = 0; i < dimension; i++) { - int index = count; + long index = count; double f = 1.0 / base[i]; int j = 0; while (index > 0) { - final int digit = scramble(i, j, base[i], index % base[i]); + final long digit = scramble(i, j, base[i], index % base[i]); v[i] += f * digit; index /= base[i]; // floor( index / base ) f /= base[i]; @@ -151,31 +159,44 @@ public double[] get() { * @param digit the j-th digit * @return the scrambled digit */ - protected int scramble(final int i, final int j, final int b, final int digit) { + protected long scramble(final int i, final int j, final int b, final long digit) { return weight != null ? (weight[i] * digit) % b : digit; } /** - * Skip to the i-th point in the Halton sequence. + * jump to the i-th point in the Halton sequence. *

* This operation can be performed in O(1). * * @param index the index in the sequence to skip to - * @return the i-th point in the Halton sequence - * @throws org.apache.commons.math4.legacy.exception.NotPositiveException NotPositiveException if index < 0 + * @return the copy of this sequence + * @throws NotPositiveException if index < 0 */ - public double[] skipTo(final int index) { + @Override + public LowDiscrepancySequence jump(final long index) throws NotPositiveException { + if(index < 0) { + throw new NotPositiveException(index); + } + HaltonSequenceGenerator copy = this.copy(); + copy.performJump(index); + return copy; + } + + /** + * do jump at the index + * @param index + */ + private void performJump(long index) { count = index; - return get(); } /** - * Returns the index i of the next point in the Halton sequence that will be returned - * by calling {@link #get()}. - * - * @return the index of the next point + * private constructor avoid side effects + * @return copy of HaltonSequenceGenerator */ - public int getNextIndex() { - return count; + private HaltonSequenceGenerator copy() { + return new HaltonSequenceGenerator(this); } + + } diff --git a/commons-math-legacy/src/main/java/org/apache/commons/math4/legacy/quasirandom/LowDiscrepancySequence.java b/commons-math-legacy/src/main/java/org/apache/commons/math4/legacy/quasirandom/LowDiscrepancySequence.java new file mode 100644 index 0000000000..00baa16ad9 --- /dev/null +++ b/commons-math-legacy/src/main/java/org/apache/commons/math4/legacy/quasirandom/LowDiscrepancySequence.java @@ -0,0 +1,32 @@ +/* + * Licensed to the Apache Software Foundation (ASF) under one or more + * contributor license agreements. See the NOTICE file distributed with + * this work for additional information regarding copyright ownership. + * The ASF licenses this file to You 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 org.apache.commons.math4.legacy.quasirandom; + +import java.util.function.Supplier; + +/** Interface to Low Discrepancy Sequence generator and supplier + * Supplier of a low discrepancy vectors + * Offers navigation through underlying sequence + */ +public interface LowDiscrepancySequence extends Supplier { + /** + * Skip to the index in the sequence + * @param index of the element to skip to + * @return T element at the index + */ + LowDiscrepancySequence jump(long index); +} diff --git a/commons-math-legacy/src/main/java/org/apache/commons/math4/legacy/random/SobolSequenceGenerator.java b/commons-math-legacy/src/main/java/org/apache/commons/math4/legacy/quasirandom/SobolSequenceGenerator.java similarity index 88% rename from commons-math-legacy/src/main/java/org/apache/commons/math4/legacy/random/SobolSequenceGenerator.java rename to commons-math-legacy/src/main/java/org/apache/commons/math4/legacy/quasirandom/SobolSequenceGenerator.java index 8269eec95c..93cda6499b 100644 --- a/commons-math-legacy/src/main/java/org/apache/commons/math4/legacy/random/SobolSequenceGenerator.java +++ b/commons-math-legacy/src/main/java/org/apache/commons/math4/legacy/quasirandom/SobolSequenceGenerator.java @@ -14,7 +14,7 @@ * See the License for the specific language governing permissions and * limitations under the License. */ -package org.apache.commons.math4.legacy.random; +package org.apache.commons.math4.legacy.quasirandom; import java.io.BufferedReader; import java.io.IOException; @@ -24,10 +24,10 @@ import java.util.Arrays; import java.util.NoSuchElementException; import java.util.StringTokenizer; -import java.util.function.Supplier; import org.apache.commons.math4.legacy.exception.MathInternalError; import org.apache.commons.math4.legacy.exception.MathParseException; +import org.apache.commons.math4.legacy.exception.NotPositiveException; import org.apache.commons.math4.legacy.exception.NotStrictlyPositiveException; import org.apache.commons.math4.legacy.exception.OutOfRangeException; import org.apache.commons.math4.legacy.core.jdkmath.AccurateMath; @@ -45,7 +45,7 @@ * The generator supports two modes: *

    *
  • sequential generation of points: {@link #get()}
  • - *
  • random access to the i-th point in the sequence: {@link #skipTo(int)}
  • + *
  • random access to the i-th point in the sequence: {@link LowDiscrepancySequence#jump(long)}
  • *
* * @see Sobol sequence (Wikipedia) @@ -53,7 +53,8 @@ * * @since 3.3 */ -public class SobolSequenceGenerator implements Supplier { +public class SobolSequenceGenerator implements LowDiscrepancySequence { + /** The number of bits to use. */ private static final int BITS = 52; @@ -73,7 +74,7 @@ public class SobolSequenceGenerator implements Supplier { private final int dimension; /** The current index in the sequence. */ - private int count; + private long count; /** The direction vector for each component. */ private final long[][] direction; @@ -171,6 +172,13 @@ public SobolSequenceGenerator(final int dimension, final InputStream is) } } + private SobolSequenceGenerator(SobolSequenceGenerator sobolSequenceGenerator) { + this.dimension = sobolSequenceGenerator.dimension; + this.count = sobolSequenceGenerator.count; + this.direction = Arrays.copyOf(sobolSequenceGenerator.direction, sobolSequenceGenerator.direction.length); + this.x = Arrays.copyOf(sobolSequenceGenerator.x, sobolSequenceGenerator.x.length); + } + /** * Load the direction vector for each dimension from the given stream. *

@@ -185,7 +193,7 @@ public SobolSequenceGenerator(final int dimension, final InputStream is) private int initFromStream(final InputStream is) throws IOException { // special case: dimension 1 -> use unit initialization for (int i = 1; i <= BITS; i++) { - direction[0][i] = 1L << (BITS - i); + direction[0][i] = 1l << (BITS - i); } final Charset charset = Charset.forName(FILE_CHARSET); @@ -261,7 +269,7 @@ public double[] get() { // find the index c of the rightmost 0 int c = 1; - int value = count - 1; + long value = count - 1; while ((value & 1) == 1) { value >>= 1; c++; @@ -281,15 +289,30 @@ public double[] get() { * This operation can be performed in O(1). * * @param index the index in the sequence to skip to - * @return the i-th point in the Sobol sequence - * @throws org.apache.commons.math4.legacy.exception.NotPositiveException NotPositiveException if index < 0 + * @return the sequence it self + * @throws NotPositiveException if index < 0 */ - public double[] skipTo(final int index) { + @Override + public LowDiscrepancySequence jump(final long index) throws NotPositiveException { + if(index < 0) { + throw new NotPositiveException(index); + } + + SobolSequenceGenerator copy = this.copy(); + copy.performJump(index); + return copy; + } + + /** + * do jump at the index + * @param index + */ + private void performJump(long index) { if (index == 0) { // reset x vector Arrays.fill(x, 0); } else { - final int i = index - 1; + final long i = index - 1; final long grayCode = i ^ (i >> 1); // compute the gray code of i = i XOR floor(i / 2) for (int j = 0; j < dimension; j++) { long result = 0; @@ -307,17 +330,14 @@ public double[] skipTo(final int index) { } } count = index; - return get(); } /** - * Returns the index i of the next point in the Sobol sequence that will be returned - * by calling {@link #get()}. - * - * @return the index of the next point + * private constructor avoid side effects + * @return copy of LowDiscrepancySequence */ - public int getNextIndex() { - return count; + private SobolSequenceGenerator copy() { + return new SobolSequenceGenerator(this); } } diff --git a/commons-math-legacy/src/main/java/org/apache/commons/math4/legacy/quasirandom/package-info.java b/commons-math-legacy/src/main/java/org/apache/commons/math4/legacy/quasirandom/package-info.java new file mode 100644 index 0000000000..f6f2ec0ad8 --- /dev/null +++ b/commons-math-legacy/src/main/java/org/apache/commons/math4/legacy/quasirandom/package-info.java @@ -0,0 +1,17 @@ +/* + * Licensed to the Apache Software Foundation (ASF) under one or more + * contributor license agreements. See the NOTICE file distributed with + * this work for additional information regarding copyright ownership. + * The ASF licenses this file to You 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 org.apache.commons.math4.legacy.quasirandom; \ No newline at end of file diff --git a/commons-math-legacy/src/test/java/org/apache/commons/math4/legacy/random/HaltonSequenceGeneratorTest.java b/commons-math-legacy/src/test/java/org/apache/commons/math4/legacy/quasirandom/HaltonSequenceGeneratorTest.java similarity index 85% rename from commons-math-legacy/src/test/java/org/apache/commons/math4/legacy/random/HaltonSequenceGeneratorTest.java rename to commons-math-legacy/src/test/java/org/apache/commons/math4/legacy/quasirandom/HaltonSequenceGeneratorTest.java index 4dd800b41b..042df96902 100644 --- a/commons-math-legacy/src/test/java/org/apache/commons/math4/legacy/random/HaltonSequenceGeneratorTest.java +++ b/commons-math-legacy/src/test/java/org/apache/commons/math4/legacy/quasirandom/HaltonSequenceGeneratorTest.java @@ -14,12 +14,13 @@ * See the License for the specific language governing permissions and * limitations under the License. */ -package org.apache.commons.math4.legacy.random; +package org.apache.commons.math4.legacy.quasirandom; -import org.junit.Assert; import org.apache.commons.math4.legacy.exception.DimensionMismatchException; +import org.apache.commons.math4.legacy.exception.NotPositiveException; import org.apache.commons.math4.legacy.exception.NullArgumentException; import org.apache.commons.math4.legacy.exception.OutOfRangeException; +import org.junit.Assert; import org.junit.Before; import org.junit.Test; @@ -63,7 +64,6 @@ public void test3DReference() { for (int i = 0; i < referenceValues.length; i++) { double[] result = generator.get(); Assert.assertArrayEquals(referenceValues[i], result, 1e-3); - Assert.assertEquals(i + 1, generator.getNextIndex()); } } @@ -73,7 +73,6 @@ public void test2DUnscrambledReference() { for (int i = 0; i < referenceValuesUnscrambled.length; i++) { double[] result = generator.get(); Assert.assertArrayEquals(referenceValuesUnscrambled[i], result, 1e-3); - Assert.assertEquals(i + 1, generator.getNextIndex()); } } @@ -119,16 +118,30 @@ public void testConstructor2() throws Exception{ } @Test - public void testSkip() { - double[] result = generator.skipTo(5); + public void testJump() { + LowDiscrepancySequence copyOfSeq = generator.jump(5); + double[] result = copyOfSeq.get(); Assert.assertArrayEquals(referenceValues[5], result, 1e-3); - Assert.assertEquals(6, generator.getNextIndex()); + for (int i = 6; i < referenceValues.length; i++) { - result = generator.get(); + result = copyOfSeq.get(); Assert.assertArrayEquals(referenceValues[i], result, 1e-3); - Assert.assertEquals(i + 1, generator.getNextIndex()); } } + + @Test(expected = NotPositiveException.class) + public void testJumpNegativeIndex() { + LowDiscrepancySequence copyOfSeq = generator.jump(-5); + } + + + + @Test + public void testFirstSupplying() { + LowDiscrepancySequence sequence = new HaltonSequenceGenerator(3); + Assert.assertArrayEquals(new double[]{0.0, 0.0, 0.0}, sequence.get(),1e-6); + } + } diff --git a/commons-math-legacy/src/test/java/org/apache/commons/math4/legacy/random/SobolSequenceGeneratorTest.java b/commons-math-legacy/src/test/java/org/apache/commons/math4/legacy/quasirandom/SobolSequenceGeneratorTest.java similarity index 82% rename from commons-math-legacy/src/test/java/org/apache/commons/math4/legacy/random/SobolSequenceGeneratorTest.java rename to commons-math-legacy/src/test/java/org/apache/commons/math4/legacy/quasirandom/SobolSequenceGeneratorTest.java index 11c45a30de..d524ccfd3f 100644 --- a/commons-math-legacy/src/test/java/org/apache/commons/math4/legacy/random/SobolSequenceGeneratorTest.java +++ b/commons-math-legacy/src/test/java/org/apache/commons/math4/legacy/quasirandom/SobolSequenceGeneratorTest.java @@ -14,16 +14,17 @@ * See the License for the specific language governing permissions and * limitations under the License. */ -package org.apache.commons.math4.legacy.random; - -import org.junit.Assert; - -import java.io.InputStream; +package org.apache.commons.math4.legacy.quasirandom; +import org.apache.commons.math4.legacy.exception.NotPositiveException; import org.apache.commons.math4.legacy.exception.OutOfRangeException; +import org.junit.Assert; import org.junit.Before; import org.junit.Test; +import java.io.InputStream; +import java.util.Arrays; + public class SobolSequenceGeneratorTest { private static final String RESOURCE_NAME = "/assets/org/apache/commons/math4/legacy/random/new-joe-kuo-6.21201"; @@ -52,7 +53,6 @@ public void test3DReference() { for (int i = 0; i < referenceValues.length; i++) { double[] result = generator.get(); Assert.assertArrayEquals(referenceValues[i], result, 1e-6); - Assert.assertEquals(i + 1, generator.getNextIndex()); } } @@ -92,16 +92,26 @@ public void testConstructor2() throws Exception{ } @Test - public void testSkip() { - double[] result = generator.skipTo(5); + public void testJump() { + LowDiscrepancySequence copyOfSeq = generator.jump(5); + double[] result = copyOfSeq.get(); Assert.assertArrayEquals(referenceValues[5], result, 1e-6); - Assert.assertEquals(6, generator.getNextIndex()); for (int i = 6; i < referenceValues.length; i++) { - result = generator.get(); + result = copyOfSeq.get(); Assert.assertArrayEquals(referenceValues[i], result, 1e-6); - Assert.assertEquals(i + 1, generator.getNextIndex()); } } + @Test(expected = NotPositiveException.class) + public void testJumpNegativeIndex() { + LowDiscrepancySequence copyOfSeq = generator.jump(-5); + } + + @Test + public void testFirstSupplying() { + LowDiscrepancySequence sequence = new SobolSequenceGenerator(3); + Assert.assertArrayEquals(new double[]{0.0, 0.0, 0.0}, sequence.get(),1e-6); + } + }