Implementation of the SYMMLQ iterative linear solver, based on Pr. Saunders FORTRAN impl.
git-svn-id: https://svn.apache.org/repos/asf/commons/proper/math/trunk@1187657 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
parent
246167dc96
commit
17f788387d
File diff suppressed because it is too large
Load Diff
|
@ -0,0 +1,641 @@
|
||||||
|
/*
|
||||||
|
* 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.math.linear;
|
||||||
|
|
||||||
|
import org.apache.commons.math.exception.DimensionMismatchException;
|
||||||
|
import org.apache.commons.math.util.FastMath;
|
||||||
|
import org.apache.commons.math.util.IterationEvent;
|
||||||
|
import org.apache.commons.math.util.IterationListener;
|
||||||
|
import org.junit.Assert;
|
||||||
|
import org.junit.Test;
|
||||||
|
|
||||||
|
public class SymmLQTest {
|
||||||
|
|
||||||
|
public void saundersTest(final int n, final boolean goodb,
|
||||||
|
final boolean precon, final double shift,
|
||||||
|
final double pertbn) {
|
||||||
|
final RealLinearOperator a = new RealLinearOperator() {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public RealVector operate(final RealVector x) {
|
||||||
|
if (x.getDimension() != n) {
|
||||||
|
throw new DimensionMismatchException(x.getDimension(), n);
|
||||||
|
}
|
||||||
|
final double[] y = new double[n];
|
||||||
|
for (int i = 0; i < n; i++) {
|
||||||
|
y[i] = (i + 1) * 1.1 / n * x.getEntry(i);
|
||||||
|
}
|
||||||
|
return new ArrayRealVector(y, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int getRowDimension() {
|
||||||
|
return n;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int getColumnDimension() {
|
||||||
|
return n;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
final double shiftm = shift;
|
||||||
|
final double pertm = FastMath.abs(pertbn);
|
||||||
|
final InvertibleRealLinearOperator m;
|
||||||
|
if (precon) {
|
||||||
|
m = new InvertibleRealLinearOperator() {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public RealVector operate(final RealVector x) {
|
||||||
|
if (x.getDimension() != n) {
|
||||||
|
throw new DimensionMismatchException(x.getDimension(),
|
||||||
|
n);
|
||||||
|
}
|
||||||
|
final double[] y = new double[n];
|
||||||
|
for (int i = 0; i < n; i++) {
|
||||||
|
double d = (i + 1) * 1.1 / n;
|
||||||
|
d = FastMath.abs(d - shiftm);
|
||||||
|
if (i % 10 == 0) {
|
||||||
|
d += pertm;
|
||||||
|
}
|
||||||
|
y[i] = d * x.getEntry(i);
|
||||||
|
}
|
||||||
|
return new ArrayRealVector(y, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int getRowDimension() {
|
||||||
|
return n;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int getColumnDimension() {
|
||||||
|
return n;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public RealVector solve(final RealVector b) {
|
||||||
|
if (b.getDimension() != n) {
|
||||||
|
throw new DimensionMismatchException(b.getDimension(),
|
||||||
|
n);
|
||||||
|
}
|
||||||
|
final double[] x = new double[n];
|
||||||
|
for (int i = 0; i < n; i++) {
|
||||||
|
double d = (i + 1) * 1.1 / n;
|
||||||
|
d = FastMath.abs(d - shiftm);
|
||||||
|
if (i % 10 == 0) {
|
||||||
|
d += pertm;
|
||||||
|
}
|
||||||
|
x[i] = b.getEntry(i) / d;
|
||||||
|
}
|
||||||
|
return new ArrayRealVector(x, false);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
} else {
|
||||||
|
m = null;
|
||||||
|
}
|
||||||
|
final RealVector xtrue = new ArrayRealVector(n);
|
||||||
|
for (int i = 0; i < n; i++) {
|
||||||
|
xtrue.setEntry(i, n - i);
|
||||||
|
}
|
||||||
|
final RealVector b = a.operate(xtrue);
|
||||||
|
b.combineToSelf(1.0, -shift, xtrue);
|
||||||
|
final SymmLQ solver = new SymmLQ(2 * n, 1E-12, true);
|
||||||
|
final RealVector x = solver.solve(a, m, b, goodb, shift);
|
||||||
|
final RealVector y = a.operate(x);
|
||||||
|
final RealVector r1 = new ArrayRealVector(n);
|
||||||
|
for (int i = 0; i < n; i++) {
|
||||||
|
final double bi = b.getEntry(i);
|
||||||
|
final double yi = y.getEntry(i);
|
||||||
|
final double xi = x.getEntry(i);
|
||||||
|
r1.setEntry(i, bi - yi + shift * xi);
|
||||||
|
}
|
||||||
|
final double enorm = x.subtract(xtrue).getNorm() / xtrue.getNorm();
|
||||||
|
final double etol = 1E-5;
|
||||||
|
Assert.assertTrue("enorm="
|
||||||
|
+ enorm
|
||||||
|
+ ", "
|
||||||
|
+ solver.getIterationManager()
|
||||||
|
.getIterations(),
|
||||||
|
enorm <= etol);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSolveSaunders1() {
|
||||||
|
saundersTest(1, false, false, 0., 0.);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSolveSaunders2() {
|
||||||
|
saundersTest(2, false, false, 0., 0.);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSolveSaunders3() {
|
||||||
|
saundersTest(1, false, true, 0., 0.);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSolveSaunders4() {
|
||||||
|
saundersTest(2, false, true, 0., 0.);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSolveSaunders5() {
|
||||||
|
saundersTest(5, false, true, 0., 0.);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSolveSaunders6() {
|
||||||
|
saundersTest(5, false, true, 0.25, 0.);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSolveSaunders7() {
|
||||||
|
saundersTest(50, false, false, 0., 0.);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSolveSaunders8() {
|
||||||
|
saundersTest(50, false, false, 0.25, 0.);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSolveSaunders9() {
|
||||||
|
saundersTest(50, false, true, 0., 0.10);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSolveSaunders10() {
|
||||||
|
saundersTest(50, false, true, 0.25, 0.10);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSolveSaunders11() {
|
||||||
|
saundersTest(1, true, false, 0., 0.);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSolveSaunders12() {
|
||||||
|
saundersTest(2, true, false, 0., 0.);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSolveSaunders13() {
|
||||||
|
saundersTest(1, true, true, 0., 0.);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSolveSaunders14() {
|
||||||
|
saundersTest(2, true, true, 0., 0.);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSolveSaunders15() {
|
||||||
|
saundersTest(5, true, true, 0., 0.);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSolveSaunders16() {
|
||||||
|
saundersTest(5, true, true, 0.25, 0.);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSolveSaunders17() {
|
||||||
|
saundersTest(50, true, false, 0., 0.);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSolveSaunders18() {
|
||||||
|
saundersTest(50, true, false, 0.25, 0.);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSolveSaunders19() {
|
||||||
|
saundersTest(50, true, true, 0., 0.10);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSolveSaunders20() {
|
||||||
|
saundersTest(50, true, true, 0.25, 0.10);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(expected = NonSquareOperatorException.class)
|
||||||
|
public void testNonSquareOperator() {
|
||||||
|
final Array2DRowRealMatrix a = new Array2DRowRealMatrix(2, 3);
|
||||||
|
final IterativeLinearSolver solver;
|
||||||
|
solver = new SymmLQ(10, 0., false);
|
||||||
|
final ArrayRealVector b = new ArrayRealVector(a.getRowDimension());
|
||||||
|
final ArrayRealVector x = new ArrayRealVector(a.getColumnDimension());
|
||||||
|
solver.solve(a, b, x);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(expected = DimensionMismatchException.class)
|
||||||
|
public void testDimensionMismatchRightHandSide() {
|
||||||
|
final Array2DRowRealMatrix a = new Array2DRowRealMatrix(3, 3);
|
||||||
|
final IterativeLinearSolver solver;
|
||||||
|
solver = new SymmLQ(10, 0., false);
|
||||||
|
final ArrayRealVector b = new ArrayRealVector(2);
|
||||||
|
solver.solve(a, b);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(expected = DimensionMismatchException.class)
|
||||||
|
public void testDimensionMismatchSolution() {
|
||||||
|
final Array2DRowRealMatrix a = new Array2DRowRealMatrix(3, 3);
|
||||||
|
final IterativeLinearSolver solver;
|
||||||
|
solver = new SymmLQ(10, 0., false);
|
||||||
|
final ArrayRealVector b = new ArrayRealVector(3);
|
||||||
|
final ArrayRealVector x = new ArrayRealVector(2);
|
||||||
|
solver.solve(a, b, x);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testUnpreconditionedSolution() {
|
||||||
|
final int n = 5;
|
||||||
|
final int maxIterations = 100;
|
||||||
|
final RealLinearOperator a = new HilbertMatrix(n);
|
||||||
|
final InverseHilbertMatrix ainv = new InverseHilbertMatrix(n);
|
||||||
|
final IterativeLinearSolver solver;
|
||||||
|
solver = new SymmLQ(maxIterations, 1E-10, true);
|
||||||
|
final RealVector b = new ArrayRealVector(n);
|
||||||
|
for (int j = 0; j < n; j++) {
|
||||||
|
b.set(0.);
|
||||||
|
b.setEntry(j, 1.);
|
||||||
|
final RealVector x = solver.solve(a, b);
|
||||||
|
for (int i = 0; i < n; i++) {
|
||||||
|
final double actual = x.getEntry(i);
|
||||||
|
final double expected = ainv.getEntry(i, j);
|
||||||
|
final double delta = 1E-6 * Math.abs(expected);
|
||||||
|
final String msg = String.format("entry[%d][%d]", i, j);
|
||||||
|
Assert.assertEquals(msg, expected, actual, delta);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testUnpreconditionedInPlaceSolutionWithInitialGuess() {
|
||||||
|
final int n = 5;
|
||||||
|
final int maxIterations = 100;
|
||||||
|
final RealLinearOperator a = new HilbertMatrix(n);
|
||||||
|
final InverseHilbertMatrix ainv = new InverseHilbertMatrix(n);
|
||||||
|
final IterativeLinearSolver solver;
|
||||||
|
solver = new SymmLQ(maxIterations, 1E-10, true);
|
||||||
|
final RealVector b = new ArrayRealVector(n);
|
||||||
|
for (int j = 0; j < n; j++) {
|
||||||
|
b.set(0.);
|
||||||
|
b.setEntry(j, 1.);
|
||||||
|
final RealVector x0 = new ArrayRealVector(n);
|
||||||
|
x0.set(1.);
|
||||||
|
final RealVector x = solver.solveInPlace(a, b, x0);
|
||||||
|
Assert.assertSame("x should be a reference to x0", x0, x);
|
||||||
|
for (int i = 0; i < n; i++) {
|
||||||
|
final double actual = x.getEntry(i);
|
||||||
|
final double expected = ainv.getEntry(i, j);
|
||||||
|
final double delta = 1E-6 * Math.abs(expected);
|
||||||
|
final String msg = String.format("entry[%d][%d)", i, j);
|
||||||
|
Assert.assertEquals(msg, expected, actual, delta);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testUnpreconditionedSolutionWithInitialGuess() {
|
||||||
|
final int n = 5;
|
||||||
|
final int maxIterations = 100;
|
||||||
|
final RealLinearOperator a = new HilbertMatrix(n);
|
||||||
|
final InverseHilbertMatrix ainv = new InverseHilbertMatrix(n);
|
||||||
|
final IterativeLinearSolver solver;
|
||||||
|
solver = new SymmLQ(maxIterations, 1E-10, true);
|
||||||
|
final RealVector b = new ArrayRealVector(n);
|
||||||
|
for (int j = 0; j < n; j++) {
|
||||||
|
b.set(0.);
|
||||||
|
b.setEntry(j, 1.);
|
||||||
|
final RealVector x0 = new ArrayRealVector(n);
|
||||||
|
x0.set(1.);
|
||||||
|
final RealVector x = solver.solve(a, b, x0);
|
||||||
|
Assert.assertNotSame("x should not be a reference to x0", x0, x);
|
||||||
|
for (int i = 0; i < n; i++) {
|
||||||
|
final double actual = x.getEntry(i);
|
||||||
|
final double expected = ainv.getEntry(i, j);
|
||||||
|
final double delta = 1E-6 * Math.abs(expected);
|
||||||
|
final String msg = String.format("entry[%d][%d]", i, j);
|
||||||
|
Assert.assertEquals(msg, expected, actual, delta);
|
||||||
|
Assert.assertEquals(msg, x0.getEntry(i), 1., Math.ulp(1.));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(expected = NonSquareOperatorException.class)
|
||||||
|
public void testNonSquarePreconditioner() {
|
||||||
|
final Array2DRowRealMatrix a = new Array2DRowRealMatrix(2, 2);
|
||||||
|
final InvertibleRealLinearOperator m;
|
||||||
|
m = new InvertibleRealLinearOperator() {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public RealVector operate(final RealVector x) {
|
||||||
|
throw new UnsupportedOperationException();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int getRowDimension() {
|
||||||
|
return 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int getColumnDimension() {
|
||||||
|
return 3;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public RealVector solve(final RealVector b) {
|
||||||
|
throw new UnsupportedOperationException();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
final PreconditionedIterativeLinearSolver solver;
|
||||||
|
solver = new SymmLQ(10, 0., false);
|
||||||
|
final ArrayRealVector b = new ArrayRealVector(a.getRowDimension());
|
||||||
|
solver.solve(a, m, b);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(expected = DimensionMismatchException.class)
|
||||||
|
public void testMismatchedOperatorDimensions() {
|
||||||
|
final Array2DRowRealMatrix a = new Array2DRowRealMatrix(2, 2);
|
||||||
|
final InvertibleRealLinearOperator m;
|
||||||
|
m = new InvertibleRealLinearOperator() {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public RealVector operate(final RealVector x) {
|
||||||
|
throw new UnsupportedOperationException();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int getRowDimension() {
|
||||||
|
return 3;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int getColumnDimension() {
|
||||||
|
return 3;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public RealVector solve(final RealVector b) {
|
||||||
|
throw new UnsupportedOperationException();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
final PreconditionedIterativeLinearSolver solver;
|
||||||
|
solver = new SymmLQ(10, 0d, false);
|
||||||
|
final ArrayRealVector b = new ArrayRealVector(a.getRowDimension());
|
||||||
|
solver.solve(a, m, b);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(expected = NonPositiveDefiniteOperatorException.class)
|
||||||
|
public void testNonPositiveDefinitePreconditioner() {
|
||||||
|
final Array2DRowRealMatrix a = new Array2DRowRealMatrix(2, 2);
|
||||||
|
a.setEntry(0, 0, 1d);
|
||||||
|
a.setEntry(0, 1, 2d);
|
||||||
|
a.setEntry(1, 0, 3d);
|
||||||
|
a.setEntry(1, 1, 4d);
|
||||||
|
final InvertibleRealLinearOperator m;
|
||||||
|
m = new InvertibleRealLinearOperator() {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public RealVector operate(final RealVector x) {
|
||||||
|
final ArrayRealVector y = new ArrayRealVector(2);
|
||||||
|
y.setEntry(0, -x.getEntry(0));
|
||||||
|
y.setEntry(1, -x.getEntry(1));
|
||||||
|
return y;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int getRowDimension() {
|
||||||
|
return 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int getColumnDimension() {
|
||||||
|
return 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public RealVector solve(final RealVector b) {
|
||||||
|
final ArrayRealVector x = new ArrayRealVector(2);
|
||||||
|
x.setEntry(0, -b.getEntry(0));
|
||||||
|
x.setEntry(1, -b.getEntry(1));
|
||||||
|
return x;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
final PreconditionedIterativeLinearSolver solver;
|
||||||
|
solver = new SymmLQ(10, 0d, true);
|
||||||
|
final ArrayRealVector b = new ArrayRealVector(2);
|
||||||
|
b.setEntry(0, -1d);
|
||||||
|
b.setEntry(1, -1d);
|
||||||
|
solver.solve(a, m, b);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testPreconditionedSolution() {
|
||||||
|
final int n = 8;
|
||||||
|
final int maxIterations = 100;
|
||||||
|
final RealLinearOperator a = new HilbertMatrix(n);
|
||||||
|
final InverseHilbertMatrix ainv = new InverseHilbertMatrix(n);
|
||||||
|
final InvertibleRealLinearOperator m = JacobiPreconditioner.create(a);
|
||||||
|
final PreconditionedIterativeLinearSolver solver;
|
||||||
|
solver = new SymmLQ(maxIterations, 1E-15, true);
|
||||||
|
final RealVector b = new ArrayRealVector(n);
|
||||||
|
for (int j = 0; j < n; j++) {
|
||||||
|
b.set(0.);
|
||||||
|
b.setEntry(j, 1.);
|
||||||
|
final RealVector x = solver.solve(a, m, b);
|
||||||
|
for (int i = 0; i < n; i++) {
|
||||||
|
final double actual = x.getEntry(i);
|
||||||
|
final double expected = ainv.getEntry(i, j);
|
||||||
|
final double delta = 1E-6 * Math.abs(expected);
|
||||||
|
final String msg = String.format("coefficient (%d, %d)", i, j);
|
||||||
|
Assert.assertEquals(msg, expected, actual, delta);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testPreconditionedSolution2() {
|
||||||
|
final int n = 100;
|
||||||
|
final int maxIterations = 100000;
|
||||||
|
final Array2DRowRealMatrix a = new Array2DRowRealMatrix(n, n);
|
||||||
|
double daux = 1.;
|
||||||
|
for (int i = 0; i < n; i++) {
|
||||||
|
a.setEntry(i, i, daux);
|
||||||
|
daux *= 1.2;
|
||||||
|
for (int j = i + 1; j < n; j++) {
|
||||||
|
if (i == j) {
|
||||||
|
} else {
|
||||||
|
final double value = 1.0;
|
||||||
|
a.setEntry(i, j, value);
|
||||||
|
a.setEntry(j, i, value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
final InvertibleRealLinearOperator m = JacobiPreconditioner.create(a);
|
||||||
|
final PreconditionedIterativeLinearSolver prec;
|
||||||
|
final IterativeLinearSolver unprec;
|
||||||
|
prec = new SymmLQ(maxIterations, 1E-15, true);
|
||||||
|
unprec = new SymmLQ(maxIterations, 1E-15, true);
|
||||||
|
final RealVector b = new ArrayRealVector(n);
|
||||||
|
final String pattern = "preconditioned SymmLQ (%d iterations) should"
|
||||||
|
+ " have been faster than unpreconditioned (%d iterations)";
|
||||||
|
String msg;
|
||||||
|
for (int j = 0; j < 1; j++) {
|
||||||
|
b.set(0.);
|
||||||
|
b.setEntry(j, 1.);
|
||||||
|
final RealVector px = prec.solve(a, m, b);
|
||||||
|
final RealVector x = unprec.solve(a, b);
|
||||||
|
final int npcg = prec.getIterationManager().getIterations();
|
||||||
|
final int ncg = unprec.getIterationManager().getIterations();
|
||||||
|
msg = String.format(pattern, npcg, ncg);
|
||||||
|
Assert.assertTrue(msg, npcg < ncg);
|
||||||
|
for (int i = 0; i < n; i++) {
|
||||||
|
msg = String.format("row %d, column %d", i, j);
|
||||||
|
final double expected = x.getEntry(i);
|
||||||
|
final double actual = px.getEntry(i);
|
||||||
|
final double delta = 5E-5 * Math.abs(expected);
|
||||||
|
Assert.assertEquals(msg, expected, actual, delta);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testEventManagement() {
|
||||||
|
final int n = 5;
|
||||||
|
final int maxIterations = 100;
|
||||||
|
final RealLinearOperator a = new HilbertMatrix(n);
|
||||||
|
final IterativeLinearSolver solver;
|
||||||
|
final int[] count = new int[] {
|
||||||
|
0, 0, 0, 0
|
||||||
|
};
|
||||||
|
final IterationListener listener = new IterationListener() {
|
||||||
|
|
||||||
|
public void initializationPerformed(final IterationEvent e) {
|
||||||
|
count[0] = 1;
|
||||||
|
count[1] = 0;
|
||||||
|
count[2] = 0;
|
||||||
|
count[3] = 0;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public void iterationPerformed(final IterationEvent e) {
|
||||||
|
++count[2];
|
||||||
|
}
|
||||||
|
|
||||||
|
public void iterationStarted(final IterationEvent e) {
|
||||||
|
++count[1];
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public void terminationPerformed(final IterationEvent e) {
|
||||||
|
++count[3];
|
||||||
|
}
|
||||||
|
};
|
||||||
|
solver = new SymmLQ(maxIterations, 1E-10, true);
|
||||||
|
solver.getIterationManager().addIterationListener(listener);
|
||||||
|
final RealVector b = new ArrayRealVector(n);
|
||||||
|
for (int j = 0; j < n; j++) {
|
||||||
|
b.set(0.);
|
||||||
|
b.setEntry(j, 1.);
|
||||||
|
solver.solve(a, b);
|
||||||
|
String msg = String.format("column %d (initialization)", j);
|
||||||
|
Assert.assertEquals(msg, 1, count[0]);
|
||||||
|
msg = String.format("column %d (iterations started)", j);
|
||||||
|
Assert.assertEquals(msg, solver.getIterationManager()
|
||||||
|
.getIterations() - 1, count[1]);
|
||||||
|
msg = String.format("column %d (iterations performed)", j);
|
||||||
|
Assert.assertEquals(msg, solver.getIterationManager()
|
||||||
|
.getIterations() - 1, count[2]);
|
||||||
|
msg = String.format("column %d (finalization)", j);
|
||||||
|
Assert.assertEquals(msg, 1, count[3]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(expected = NonSelfAdjointOperatorException.class)
|
||||||
|
public void testNonSelfAdjointOperator() {
|
||||||
|
final RealLinearOperator a;
|
||||||
|
a = new Array2DRowRealMatrix(new double[][] {
|
||||||
|
{
|
||||||
|
1., 2., 3.
|
||||||
|
}, {
|
||||||
|
2., 4., 5.
|
||||||
|
}, {
|
||||||
|
2.999, 5., 6.
|
||||||
|
}
|
||||||
|
});
|
||||||
|
final RealVector b;
|
||||||
|
b = new ArrayRealVector(new double[] {
|
||||||
|
1., 1., 1.
|
||||||
|
});
|
||||||
|
new SymmLQ(100, 1., true).solve(a, b);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(expected = NonSelfAdjointOperatorException.class)
|
||||||
|
public void testNonSelfAdjointPreconditioner() {
|
||||||
|
final RealLinearOperator a = new Array2DRowRealMatrix(new double[][] {
|
||||||
|
{
|
||||||
|
1., 2., 3.
|
||||||
|
}, {
|
||||||
|
2., 4., 5.
|
||||||
|
}, {
|
||||||
|
3., 5., 6.
|
||||||
|
}
|
||||||
|
});
|
||||||
|
final Array2DRowRealMatrix mMat;
|
||||||
|
mMat = new Array2DRowRealMatrix(new double[][] {
|
||||||
|
{
|
||||||
|
1., 0., 1.
|
||||||
|
}, {
|
||||||
|
0., 1., 0.
|
||||||
|
}, {
|
||||||
|
0., 0., 1.
|
||||||
|
}
|
||||||
|
});
|
||||||
|
final DecompositionSolver mSolver;
|
||||||
|
mSolver = new LUDecomposition(mMat).getSolver();
|
||||||
|
final InvertibleRealLinearOperator m;
|
||||||
|
m = new InvertibleRealLinearOperator() {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public RealVector operate(final RealVector x) {
|
||||||
|
return mMat.operate(x);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int getRowDimension() {
|
||||||
|
return mMat.getRowDimension();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int getColumnDimension() {
|
||||||
|
return mMat.getColumnDimension();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public RealVector solve(final RealVector b) {
|
||||||
|
return mSolver.solve(b);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
final RealVector b = new ArrayRealVector(new double[] {
|
||||||
|
1., 1., 1.
|
||||||
|
});
|
||||||
|
new SymmLQ(100, 1., true).solve(a, m, b);
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue