MATH-1027
Added accessors and (lexicographic) comparator. git-svn-id: https://svn.apache.org/repos/asf/commons/proper/math/trunk@1520605 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
parent
928a556123
commit
d3b4651afb
|
@ -17,8 +17,13 @@
|
||||||
package org.apache.commons.math3.util;
|
package org.apache.commons.math3.util;
|
||||||
|
|
||||||
import java.util.Iterator;
|
import java.util.Iterator;
|
||||||
|
import java.util.Comparator;
|
||||||
|
import java.util.Arrays;
|
||||||
import java.util.NoSuchElementException;
|
import java.util.NoSuchElementException;
|
||||||
|
import java.io.Serializable;
|
||||||
import org.apache.commons.math3.exception.MathInternalError;
|
import org.apache.commons.math3.exception.MathInternalError;
|
||||||
|
import org.apache.commons.math3.exception.DimensionMismatchException;
|
||||||
|
import org.apache.commons.math3.exception.OutOfRangeException;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Utility to create <a href="http://en.wikipedia.org/wiki/Combination">
|
* Utility to create <a href="http://en.wikipedia.org/wiki/Combination">
|
||||||
|
@ -96,6 +101,24 @@ public class Combinations implements Iterable<int[]> {
|
||||||
this.iterationOrder = iterationOrder;
|
this.iterationOrder = iterationOrder;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Gets the size of the set from which combinations are drawn.
|
||||||
|
*
|
||||||
|
* @return the size of the universe.
|
||||||
|
*/
|
||||||
|
public int getN() {
|
||||||
|
return n;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Gets the number of elements in each combination.
|
||||||
|
*
|
||||||
|
* @return the size of the subsets to be enumerated.
|
||||||
|
*/
|
||||||
|
public int getK() {
|
||||||
|
return k;
|
||||||
|
}
|
||||||
|
|
||||||
/** {@inheritDoc} */
|
/** {@inheritDoc} */
|
||||||
@Override
|
@Override
|
||||||
public Iterator<int[]> iterator() {
|
public Iterator<int[]> iterator() {
|
||||||
|
@ -119,6 +142,24 @@ public class Combinations implements Iterable<int[]> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Defines a lexicographic ordering of combinations.
|
||||||
|
* The returned comparator allows to compare any two combinations
|
||||||
|
* that can be produced by this instance's {@link #iterator() iterator}.
|
||||||
|
* Its {@code compare(int[],int[])} method will throw exceptions if
|
||||||
|
* passed combinations that are inconsistent with this instance:
|
||||||
|
* <ul>
|
||||||
|
* <li>{@code DimensionMismatchException} if the array lengths are not
|
||||||
|
* equal to {@code k},</li>
|
||||||
|
* <li>{@code OutOfRangeException} if an element of the array is not
|
||||||
|
* within the interval [0, {@code n}).</li>
|
||||||
|
* </ul>
|
||||||
|
* @return a lexicographic comparator.
|
||||||
|
*/
|
||||||
|
public Comparator<int[]> comparator() {
|
||||||
|
return new LexicographicComparator(n, k);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Lexicographic combinations iterator.
|
* Lexicographic combinations iterator.
|
||||||
* <p>
|
* <p>
|
||||||
|
@ -278,4 +319,89 @@ public class Combinations implements Iterable<int[]> {
|
||||||
throw new UnsupportedOperationException();
|
throw new UnsupportedOperationException();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Defines the lexicographic ordering of combinations, using
|
||||||
|
* the {@link #lexNorm(int[],int)} method.
|
||||||
|
*/
|
||||||
|
private static class LexicographicComparator
|
||||||
|
implements Comparator<int[]>, Serializable {
|
||||||
|
/** Serializable version identifier. */
|
||||||
|
private static final long serialVersionUID = 20130906L;
|
||||||
|
/** Size of the set from which combinations are drawn. */
|
||||||
|
private final int n;
|
||||||
|
/** Number of elements in each combination. */
|
||||||
|
private final int k;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param n Size of the set from which subsets are selected.
|
||||||
|
* @param k Size of the subsets to be enumerated.
|
||||||
|
*/
|
||||||
|
public LexicographicComparator(int n,
|
||||||
|
int k) {
|
||||||
|
this.n = n;
|
||||||
|
this.k = k;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* {@inheritDoc}
|
||||||
|
*
|
||||||
|
* @throws DimensionMismatchException if the array lengths are not
|
||||||
|
* equal to {@code k}.
|
||||||
|
* @throws OutOfRangeException if an element of the array is not
|
||||||
|
* within the interval [0, {@code n}).
|
||||||
|
*/
|
||||||
|
public int compare(int[] c1,
|
||||||
|
int[] c2) {
|
||||||
|
if (c1.length != k) {
|
||||||
|
throw new DimensionMismatchException(c1.length, k);
|
||||||
|
}
|
||||||
|
if (c2.length != k) {
|
||||||
|
throw new DimensionMismatchException(c2.length, k);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Method "lexNorm" works with ordered arrays.
|
||||||
|
final int[] c1s = MathArrays.copyOf(c1);
|
||||||
|
Arrays.sort(c1s);
|
||||||
|
final int[] c2s = MathArrays.copyOf(c2);
|
||||||
|
Arrays.sort(c2s);
|
||||||
|
|
||||||
|
final long v1 = lexNorm(c1s);
|
||||||
|
final long v2 = lexNorm(c2s);
|
||||||
|
|
||||||
|
if (v1 < v2) {
|
||||||
|
return -1;
|
||||||
|
} else if (v1 > v2) {
|
||||||
|
return 1;
|
||||||
|
} else {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Computes the value (in base 10) represented by the digit
|
||||||
|
* (interpreted in base {@code n}) in the input array in reverse
|
||||||
|
* order.
|
||||||
|
* For example if {@code c} is {@code {3, 2, 1}}, and {@code n}
|
||||||
|
* is 3, the method will return 18.
|
||||||
|
*
|
||||||
|
* @param c Input array.
|
||||||
|
* @return the lexicographic norm.
|
||||||
|
* @throws OutOfRangeException if an element of the array is not
|
||||||
|
* within the interval [0, {@code n}).
|
||||||
|
*/
|
||||||
|
private long lexNorm(int[] c) {
|
||||||
|
long ret = 0;
|
||||||
|
for (int i = 0; i < c.length; i++) {
|
||||||
|
final int digit = c[i];
|
||||||
|
if (digit < 0 ||
|
||||||
|
digit >= n) {
|
||||||
|
throw new OutOfRangeException(digit, 0, n - 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
ret += c[i] * ArithmeticUtils.pow(n, (long) i);
|
||||||
|
}
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -17,6 +17,9 @@
|
||||||
package org.apache.commons.math3.util;
|
package org.apache.commons.math3.util;
|
||||||
|
|
||||||
import java.util.Iterator;
|
import java.util.Iterator;
|
||||||
|
import java.util.Comparator;
|
||||||
|
import org.apache.commons.math3.exception.DimensionMismatchException;
|
||||||
|
import org.apache.commons.math3.exception.OutOfRangeException;
|
||||||
import org.apache.commons.math3.exception.MathIllegalArgumentException;
|
import org.apache.commons.math3.exception.MathIllegalArgumentException;
|
||||||
import org.junit.Assert;
|
import org.junit.Assert;
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
|
@ -26,19 +29,93 @@ import org.junit.Test;
|
||||||
*
|
*
|
||||||
* @version $Id$
|
* @version $Id$
|
||||||
*/
|
*/
|
||||||
public class CombinationsTest {
|
public class CombinationsTest {
|
||||||
|
@Test
|
||||||
|
public void testAccessor1() {
|
||||||
|
final int n = 5;
|
||||||
|
final int k = 3;
|
||||||
|
Assert.assertEquals(n, new Combinations(n, k).getN());
|
||||||
|
}
|
||||||
|
@Test
|
||||||
|
public void testAccessor2() {
|
||||||
|
final int n = 5;
|
||||||
|
final int k = 3;
|
||||||
|
Assert.assertEquals(k, new Combinations(n, k).getK());
|
||||||
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testLexicographicIterator() {
|
public void testLexicographicIterator() {
|
||||||
checkLexicographicIterator(new Combinations(5, 3), 5, 3);
|
checkLexicographicIterator(new Combinations(5, 3));
|
||||||
checkLexicographicIterator(new Combinations(6, 4), 6, 4);
|
checkLexicographicIterator(new Combinations(6, 4));
|
||||||
checkLexicographicIterator(new Combinations(8, 2), 8, 2);
|
checkLexicographicIterator(new Combinations(8, 2));
|
||||||
checkLexicographicIterator(new Combinations(6, 1), 6, 1);
|
checkLexicographicIterator(new Combinations(6, 1));
|
||||||
checkLexicographicIterator(new Combinations(3, 3), 3, 3);
|
checkLexicographicIterator(new Combinations(3, 3));
|
||||||
checkLexicographicIterator(new Combinations(1, 1), 1, 1);
|
checkLexicographicIterator(new Combinations(1, 1));
|
||||||
checkLexicographicIterator(new Combinations(1, 0), 1, 0);
|
checkLexicographicIterator(new Combinations(1, 0));
|
||||||
checkLexicographicIterator(new Combinations(0, 0), 0, 0);
|
checkLexicographicIterator(new Combinations(0, 0));
|
||||||
checkLexicographicIterator(new Combinations(4, 2), 4, 2);
|
checkLexicographicIterator(new Combinations(4, 2));
|
||||||
checkLexicographicIterator(new Combinations(123, 2), 123, 2);
|
checkLexicographicIterator(new Combinations(123, 2));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(expected=DimensionMismatchException.class)
|
||||||
|
public void testLexicographicComparatorWrongIterate1() {
|
||||||
|
final int n = 5;
|
||||||
|
final int k = 3;
|
||||||
|
final Comparator<int[]> comp = new Combinations(n, k).comparator();
|
||||||
|
comp.compare(new int[] {1}, new int[] {0, 1, 2});
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(expected=DimensionMismatchException.class)
|
||||||
|
public void testLexicographicComparatorWrongIterate2() {
|
||||||
|
final int n = 5;
|
||||||
|
final int k = 3;
|
||||||
|
final Comparator<int[]> comp = new Combinations(n, k).comparator();
|
||||||
|
comp.compare(new int[] {0, 1, 2}, new int[] {0, 1, 2, 3});
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(expected=OutOfRangeException.class)
|
||||||
|
public void testLexicographicComparatorWrongIterate3() {
|
||||||
|
final int n = 5;
|
||||||
|
final int k = 3;
|
||||||
|
final Comparator<int[]> comp = new Combinations(n, k).comparator();
|
||||||
|
comp.compare(new int[] {1, 2, 5}, new int[] {0, 1, 2});
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(expected=OutOfRangeException.class)
|
||||||
|
public void testLexicographicComparatorWrongIterate4() {
|
||||||
|
final int n = 5;
|
||||||
|
final int k = 3;
|
||||||
|
final Comparator<int[]> comp = new Combinations(n, k).comparator();
|
||||||
|
comp.compare(new int[] {1, 2, 4}, new int[] {-1, 1, 2});
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testLexicographicComparator() {
|
||||||
|
final int n = 5;
|
||||||
|
final int k = 3;
|
||||||
|
final Comparator<int[]> comp = new Combinations(n, k).comparator();
|
||||||
|
Assert.assertEquals(1, comp.compare(new int[] {1, 2, 4},
|
||||||
|
new int[] {1, 2, 3}));
|
||||||
|
Assert.assertEquals(-1, comp.compare(new int[] {0, 1, 4},
|
||||||
|
new int[] {0, 2, 4}));
|
||||||
|
Assert.assertEquals(0, comp.compare(new int[] {1, 3, 4},
|
||||||
|
new int[] {1, 3, 4}));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Check that iterates can be passed unsorted.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void testLexicographicComparatorUnsorted() {
|
||||||
|
final int n = 5;
|
||||||
|
final int k = 3;
|
||||||
|
final Comparator<int[]> comp = new Combinations(n, k).comparator();
|
||||||
|
Assert.assertEquals(1, comp.compare(new int[] {1, 4, 2},
|
||||||
|
new int[] {1, 3, 2}));
|
||||||
|
Assert.assertEquals(-1, comp.compare(new int[] {0, 4, 1},
|
||||||
|
new int[] {0, 4, 2}));
|
||||||
|
Assert.assertEquals(0, comp.compare(new int[] {1, 4, 3},
|
||||||
|
new int[] {1, 3, 4}));
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
|
@ -71,25 +148,35 @@ public class CombinationsTest {
|
||||||
* and each array itself increasing.
|
* and each array itself increasing.
|
||||||
*
|
*
|
||||||
* @param c Combinations.
|
* @param c Combinations.
|
||||||
* @param n Size of universe.
|
|
||||||
* @param k Size of subsets.
|
|
||||||
*/
|
*/
|
||||||
private void checkLexicographicIterator(Iterable<int[]> c,
|
private void checkLexicographicIterator(Combinations c) {
|
||||||
int n,
|
final Comparator<int[]> comp = c.comparator();
|
||||||
int k) {
|
final int n = c.getN();
|
||||||
long lastLex = -1;
|
final int k = c.getK();
|
||||||
long length = 0;
|
|
||||||
|
int[] lastIterate = null;
|
||||||
|
|
||||||
|
long numIterates = 0;
|
||||||
for (int[] iterate : c) {
|
for (int[] iterate : c) {
|
||||||
Assert.assertEquals(k, iterate.length);
|
Assert.assertEquals(k, iterate.length);
|
||||||
final long curLex = lexNorm(iterate, n);
|
|
||||||
Assert.assertTrue(curLex > lastLex);
|
// Check that the sequence of iterates is ordered.
|
||||||
lastLex = curLex;
|
if (lastIterate != null) {
|
||||||
length++;
|
Assert.assertTrue(comp.compare(iterate, lastIterate) == 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check that each iterate is ordered.
|
||||||
for (int i = 1; i < iterate.length; i++) {
|
for (int i = 1; i < iterate.length; i++) {
|
||||||
Assert.assertTrue(iterate[i] > iterate[i - 1]);
|
Assert.assertTrue(iterate[i] > iterate[i - 1]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
lastIterate = iterate;
|
||||||
|
++numIterates;
|
||||||
}
|
}
|
||||||
Assert.assertEquals(CombinatoricsUtils.binomialCoefficient(n, k), length);
|
|
||||||
|
// Check the number of iterates.
|
||||||
|
Assert.assertEquals(CombinatoricsUtils.binomialCoefficient(n, k),
|
||||||
|
numIterates);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
|
@ -108,20 +195,4 @@ public class CombinationsTest {
|
||||||
// ignored
|
// ignored
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Returns the value represented by the digits in the input array in reverse order.
|
|
||||||
* For example [3,2,1] returns 123.
|
|
||||||
*
|
|
||||||
* @param iterate input array
|
|
||||||
* @param n size of universe
|
|
||||||
* @return lexicographic norm
|
|
||||||
*/
|
|
||||||
private long lexNorm(int[] iterate, int n) {
|
|
||||||
long ret = 0;
|
|
||||||
for (int i = iterate.length - 1; i >= 0; i--) {
|
|
||||||
ret += iterate[i] * ArithmeticUtils.pow(n, (long) i);
|
|
||||||
}
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue