[MATH-842] Fix Blands rule by applying it also to pivot column selection, add getter/setter for cycle prevention, reduce default cut-off threshold to 1e-10.
git-svn-id: https://svn.apache.org/repos/asf/commons/proper/math/trunk@1523495 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
parent
fb6f2aec92
commit
a6925e3e8a
|
@ -47,6 +47,9 @@ public class SimplexSolver extends AbstractLinearOptimizer {
|
||||||
/** Amount of error to accept in floating point comparisons (as ulps). */
|
/** Amount of error to accept in floating point comparisons (as ulps). */
|
||||||
private final int maxUlps;
|
private final int maxUlps;
|
||||||
|
|
||||||
|
/** Prevent cycles via Bland's rule. */
|
||||||
|
private boolean cyclePrevention;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Build a simplex solver with default settings.
|
* Build a simplex solver with default settings.
|
||||||
*/
|
*/
|
||||||
|
@ -62,6 +65,33 @@ public class SimplexSolver extends AbstractLinearOptimizer {
|
||||||
public SimplexSolver(final double epsilon, final int maxUlps) {
|
public SimplexSolver(final double epsilon, final int maxUlps) {
|
||||||
this.epsilon = epsilon;
|
this.epsilon = epsilon;
|
||||||
this.maxUlps = maxUlps;
|
this.maxUlps = maxUlps;
|
||||||
|
this.cyclePrevention = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Enables/disables cycle prevention using Bland's rule.
|
||||||
|
* <p>
|
||||||
|
* In case of a degeneracy the simplex algorithm might cycle. This can be prevented
|
||||||
|
* by using Bland's pivot selection rule. The drawback of this rule is that it may result
|
||||||
|
* in pivot choices that do not significantly improve the objective function value,
|
||||||
|
* thus increasing the number of required iterations.
|
||||||
|
* <p>
|
||||||
|
* By default, the cycle prevention is enabled.
|
||||||
|
*
|
||||||
|
* @param cyclePrevention if cycle prevention shall be used
|
||||||
|
*
|
||||||
|
* @see <a href="http://www.stanford.edu/class/msande310/blandrule.pdf">Bland's rule</a>
|
||||||
|
*/
|
||||||
|
public void setCyclePrevention(boolean cyclePrevention) {
|
||||||
|
this.cyclePrevention = cyclePrevention;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns whether cycle prevention (using Bland's rule) is enabled.
|
||||||
|
* @return {@code true} if cycle prevention is enable, {@code false} otherwise
|
||||||
|
*/
|
||||||
|
public boolean getCyclePrevention() {
|
||||||
|
return cyclePrevention;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -79,14 +109,42 @@ public class SimplexSolver extends AbstractLinearOptimizer {
|
||||||
if (entry < minValue) {
|
if (entry < minValue) {
|
||||||
minValue = entry;
|
minValue = entry;
|
||||||
minPos = i;
|
minPos = i;
|
||||||
|
|
||||||
|
// Bland's rule: chose the entering column with the lowest index
|
||||||
|
if (cyclePrevention && isValidPivotColumn(tableau, i)) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return minPos;
|
return minPos;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checks whether the given column is valid pivot column, i.e. will result
|
||||||
|
* in a valid pivot row.
|
||||||
|
* <p>
|
||||||
|
* When applying Bland's rule to select the pivot column, it may happen that
|
||||||
|
* there is no corresponding pivot row. This method will check if the selected
|
||||||
|
* pivot column will return a valid pivot row.
|
||||||
|
*
|
||||||
|
* @param tableau simplex tableau for the problem
|
||||||
|
* @param col the column to test
|
||||||
|
* @return {@code true} if the pivot column is valid, {@code false} otherwise
|
||||||
|
*/
|
||||||
|
private boolean isValidPivotColumn(SimplexTableau tableau, int col) {
|
||||||
|
for (int i = tableau.getNumObjectiveFunctions(); i < tableau.getHeight(); i++) {
|
||||||
|
final double entry = tableau.getEntry(i, col);
|
||||||
|
|
||||||
|
if (Precision.compareTo(entry, 0d, maxUlps) > 0) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns the row with the minimum ratio as given by the minimum ratio test (MRT).
|
* Returns the row with the minimum ratio as given by the minimum ratio test (MRT).
|
||||||
* @param tableau simple tableau for the problem
|
* @param tableau simplex tableau for the problem
|
||||||
* @param col the column to test the ratio of. See {@link #getPivotColumn(SimplexTableau)}
|
* @param col the column to test the ratio of. See {@link #getPivotColumn(SimplexTableau)}
|
||||||
* @return row with the minimum ratio
|
* @return row with the minimum ratio
|
||||||
*/
|
*/
|
||||||
|
@ -136,11 +194,7 @@ public class SimplexSolver extends AbstractLinearOptimizer {
|
||||||
//
|
//
|
||||||
// see http://www.stanford.edu/class/msande310/blandrule.pdf
|
// see http://www.stanford.edu/class/msande310/blandrule.pdf
|
||||||
// see http://en.wikipedia.org/wiki/Bland%27s_rule (not equivalent to the above paper)
|
// see http://en.wikipedia.org/wiki/Bland%27s_rule (not equivalent to the above paper)
|
||||||
//
|
if (cyclePrevention) {
|
||||||
// Additional heuristic: if we did not get a solution after half of maxIterations
|
|
||||||
// revert to the simple case of just returning the top-most row
|
|
||||||
// This heuristic is based on empirical data gathered while investigating MATH-828.
|
|
||||||
if (getIterations() < getMaxIterations() / 2) {
|
|
||||||
Integer minRow = null;
|
Integer minRow = null;
|
||||||
int minIndex = tableau.getWidth();
|
int minIndex = tableau.getWidth();
|
||||||
final int varStart = tableau.getNumObjectiveFunctions();
|
final int varStart = tableau.getNumObjectiveFunctions();
|
||||||
|
|
|
@ -73,7 +73,7 @@ class SimplexTableau implements Serializable {
|
||||||
private static final int DEFAULT_ULPS = 10;
|
private static final int DEFAULT_ULPS = 10;
|
||||||
|
|
||||||
/** The cut-off threshold to zero-out entries. */
|
/** The cut-off threshold to zero-out entries. */
|
||||||
private static final double CUTOFF_THRESHOLD = 1e-12;
|
private static final double CUTOFF_THRESHOLD = 1e-10;
|
||||||
|
|
||||||
/** Serializable version identifier. */
|
/** Serializable version identifier. */
|
||||||
private static final long serialVersionUID = -1369660067587938365L;
|
private static final long serialVersionUID = -1369660067587938365L;
|
||||||
|
|
|
@ -30,6 +30,31 @@ import org.junit.Test;
|
||||||
|
|
||||||
public class SimplexSolverTest {
|
public class SimplexSolverTest {
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testMath842Cycle() {
|
||||||
|
// from http://www.math.toronto.edu/mpugh/Teaching/APM236_04/bland
|
||||||
|
// maximize 10 x1 - 57 x2 - 9 x3 - 24 x4
|
||||||
|
// subject to
|
||||||
|
// 1/2 x1 - 11/2 x2 - 5/2 x3 + 9 x4 <= 0
|
||||||
|
// 1/2 x1 - 3/2 x2 - 1/2 x3 + x4 <= 0
|
||||||
|
// x1 <= 1
|
||||||
|
// x1,x2,x3,x4 >= 0
|
||||||
|
|
||||||
|
LinearObjectiveFunction f = new LinearObjectiveFunction(new double[] { 10, -57, -9, -24}, 0);
|
||||||
|
|
||||||
|
ArrayList <LinearConstraint>constraints = new ArrayList<LinearConstraint>();
|
||||||
|
|
||||||
|
constraints.add(new LinearConstraint(new double[] {0.5, -5.5, -2.5, 9}, Relationship.LEQ, 0));
|
||||||
|
constraints.add(new LinearConstraint(new double[] {0.5, -1.5, -0.5, 1}, Relationship.LEQ, 0));
|
||||||
|
constraints.add(new LinearConstraint(new double[] { 1, 0, 0, 0}, Relationship.LEQ, 1));
|
||||||
|
|
||||||
|
double epsilon = 1e-6;
|
||||||
|
SimplexSolver solver = new SimplexSolver();
|
||||||
|
PointValuePair solution = solver.optimize(f, constraints, GoalType.MAXIMIZE, true);
|
||||||
|
Assert.assertEquals(1.0d, solution.getValue(), epsilon);
|
||||||
|
Assert.assertTrue(validSolution(solution, constraints, epsilon));
|
||||||
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testMath828() {
|
public void testMath828() {
|
||||||
LinearObjectiveFunction f = new LinearObjectiveFunction(
|
LinearObjectiveFunction f = new LinearObjectiveFunction(
|
||||||
|
|
Loading…
Reference in New Issue