001/* 002 * BioJava development code 003 * 004 * This code may be freely distributed and modified under the 005 * terms of the GNU Lesser General Public Licence. This should 006 * be distributed with the code. If you do not have a copy, 007 * see: 008 * 009 * http://www.gnu.org/copyleft/lesser.html 010 * 011 * Copyright for this code is held jointly by the individual 012 * authors. These should be listed in @author doc comments. 013 * 014 * For more information on the BioJava project and its aims, 015 * or to join the biojava-l mailing list, visit the home page 016 * at: 017 * 018 * http://www.biojava.org/ 019 * 020 */ 021/* Derived from the MathWorks and NIST implementation, which was released into 022 * the public domain. 023 * 024 * http://math.nist.gov/javanumerics/jama/ 025 */ 026package org.biojava.nbio.structure.jama; 027 028import java.io.BufferedReader; 029import java.io.PrintWriter; 030import java.io.StreamTokenizer; 031import java.io.StringWriter; 032import java.text.DecimalFormat; 033import java.text.DecimalFormatSymbols; 034import java.text.NumberFormat; 035import java.util.Locale; 036 037 038/** 039 * Jama = Java Matrix class. 040 * <P> 041 * The Java Matrix Class provides the fundamental operations of numerical 042 * linear algebra. Various constructors create Matrices from two dimensional 043 * arrays of double precision floating point numbers. Various "gets" and 044 * "sets" provide access to submatrices and matrix elements. Several methods 045 * implement basic matrix arithmetic, including matrix addition and 046 * multiplication, matrix norms, and element-by-element array operations. 047 * Methods for reading and printing matrices are also included. All the 048 * operations in this version of the Matrix Class involve real matrices. 049 * Complex matrices may be handled in a future version. 050 * <P> 051 * Five fundamental matrix decompositions, which consist of pairs or triples 052 * of matrices, permutation vectors, and the like, produce results in five 053 * decomposition classes. These decompositions are accessed by the Matrix 054 * class to compute solutions of simultaneous linear equations, determinants, 055 * inverses and other matrix functions. The five decompositions are: 056 * <UL> 057 * <LI>Cholesky Decomposition of symmetric, positive definite matrices. 058 * <LI>LU Decomposition of rectangular matrices. 059 * <LI>QR Decomposition of rectangular matrices. 060 * <LI>Singular Value Decomposition of rectangular matrices. 061 * <LI>Eigenvalue Decomposition of both symmetric and nonsymmetric square matrices. 062 * </UL> 063 * <DL> 064 * <DT><B>Example of use:</B></DT> 065 * 066 * <DD>Solve a linear system A x = b and compute the residual norm, ||b - A x||. 067 * <P><PRE> 068 * double[][] vals = {{1.,2.,3},{4.,5.,6.},{7.,8.,10.}}; 069 * Matrix A = new Matrix(vals); 070 * Matrix b = Matrix.random(3,1); 071 * Matrix x = A.solve(b); 072 * Matrix r = A.times(x).minus(b); 073 * double rnorm = r.normInf(); 074 * </PRE></DD> 075 * </DL> 076 * 077 * @author The MathWorks, Inc. and the National Institute of Standards and Technology. 078 * @version 5 August 1998 079 */ 080 081public class Matrix implements Cloneable, java.io.Serializable { 082 083 static final long serialVersionUID = 8492558293015348719l; 084 085 086/* ------------------------ 087 Class variables 088 * ------------------------ */ 089 090 /** Array for internal storage of elements. 091 @serial internal array storage. 092 */ 093 private double[][] A; 094 095 /** Row and column dimensions. 096 @serial row dimension. 097 @serial column dimension. 098 */ 099 private int m, n; 100 101/* ------------------------ 102 Constructors 103 * ------------------------ */ 104 105 /** Construct an m-by-n matrix of zeros. 106 @param m Number of rows. 107 @param n Number of colums. 108 */ 109 110 public Matrix (int m, int n) { 111 this.m = m; 112 this.n = n; 113 A = new double[m][n]; 114 } 115 116 /** Construct an m-by-n constant matrix. 117 @param m Number of rows. 118 @param n Number of colums. 119 @param s Fill the matrix with this scalar value. 120 */ 121 122 public Matrix (int m, int n, double s) { 123 this.m = m; 124 this.n = n; 125 A = new double[m][n]; 126 for (int i = 0; i < m; i++) { 127 for (int j = 0; j < n; j++) { 128 A[i][j] = s; 129 } 130 } 131 } 132 133 /** Construct a matrix from a 2-D array. 134 @param A Two-dimensional array of doubles. 135 @exception IllegalArgumentException All rows must have the same length 136 @see #constructWithCopy 137 */ 138 139 public Matrix (double[][] A) { 140 m = A.length; 141 n = A[0].length; 142 for (int i = 0; i < m; i++) { 143 if (A[i].length != n) { 144 throw new IllegalArgumentException("All rows must have the same length."); 145 } 146 } 147 this.A = A; 148 } 149 150 /** Construct a matrix quickly without checking arguments. 151 @param A Two-dimensional array of doubles. 152 @param m Number of rows. 153 @param n Number of colums. 154 */ 155 156 public Matrix (double[][] A, int m, int n) { 157 this.A = A; 158 this.m = m; 159 this.n = n; 160 } 161 162 /** Construct a matrix from a one-dimensional packed array 163 @param vals One-dimensional array of doubles, packed by columns (ala Fortran). 164 @param m Number of rows. 165 @exception IllegalArgumentException Array length must be a multiple of m. 166 */ 167 168 public Matrix (double[] vals, int m) { 169 this.m = m; 170 n = (m != 0 ? vals.length/m : 0); 171 if (m*n != vals.length) { 172 throw new IllegalArgumentException("Array length must be a multiple of m."); 173 } 174 A = new double[m][n]; 175 for (int i = 0; i < m; i++) { 176 for (int j = 0; j < n; j++) { 177 A[i][j] = vals[i+j*m]; 178 } 179 } 180 } 181 182/* ------------------------ 183 Public Methods 184 * ------------------------ */ 185 186 /** Construct a matrix from a copy of a 2-D array. 187 @param A Two-dimensional array of doubles. 188 @exception IllegalArgumentException All rows must have the same length 189 @return a Matrix 190 */ 191 192 public static Matrix constructWithCopy(double[][] A) { 193 int m = A.length; 194 int n = A[0].length; 195 Matrix X = new Matrix(m,n); 196 double[][] C = X.getArray(); 197 for (int i = 0; i < m; i++) { 198 if (A[i].length != n) { 199 throw new IllegalArgumentException 200 ("All rows must have the same length."); 201 } 202 for (int j = 0; j < n; j++) { 203 C[i][j] = A[i][j]; 204 } 205 } 206 return X; 207 } 208 209 /** Make a deep copy of a matrix 210 * @return a identical copy of the Matrix 211 */ 212 213 public Matrix copy () { 214 Matrix X = new Matrix(m,n); 215 double[][] C = X.getArray(); 216 for (int i = 0; i < m; i++) { 217 for (int j = 0; j < n; j++) { 218 C[i][j] = A[i][j]; 219 } 220 } 221 return X; 222 } 223 224 /** Clone the Matrix object. 225 */ 226 227 @Override 228public Object clone () { 229 return this.copy(); 230 } 231 232 /** Access the internal two-dimensional array. 233 @return Pointer to the two-dimensional array of matrix elements. 234 */ 235 236 public double[][] getArray () { 237 return A; 238 } 239 240 /** Copy the internal two-dimensional array. 241 @return Two-dimensional array copy of matrix elements. 242 */ 243 244 public double[][] getArrayCopy () { 245 double[][] C = new double[m][n]; 246 for (int i = 0; i < m; i++) { 247 for (int j = 0; j < n; j++) { 248 C[i][j] = A[i][j]; 249 } 250 } 251 return C; 252 } 253 254 /** Make a one-dimensional column packed copy of the internal array. 255 @return Matrix elements packed in a one-dimensional array by columns. 256 */ 257 258 public double[] getColumnPackedCopy () { 259 double[] vals = new double[m*n]; 260 for (int i = 0; i < m; i++) { 261 for (int j = 0; j < n; j++) { 262 vals[i+j*m] = A[i][j]; 263 } 264 } 265 return vals; 266 } 267 268 /** Make a one-dimensional row packed copy of the internal array. 269 @return Matrix elements packed in a one-dimensional array by rows. 270 */ 271 272 public double[] getRowPackedCopy () { 273 double[] vals = new double[m*n]; 274 for (int i = 0; i < m; i++) { 275 for (int j = 0; j < n; j++) { 276 vals[i*n+j] = A[i][j]; 277 } 278 } 279 return vals; 280 } 281 282 /** Get row dimension. 283 @return m, the number of rows. 284 */ 285 286 public int getRowDimension () { 287 return m; 288 } 289 290 /** Get column dimension. 291 @return n, the number of columns. 292 */ 293 294 public int getColumnDimension () { 295 return n; 296 } 297 298 /** Get a single element. 299 @param i Row index. 300 @param j Column index. 301 @return A(i,j) 302 @exception ArrayIndexOutOfBoundsException 303 */ 304 305 public double get (int i, int j) { 306 return A[i][j]; 307 } 308 309 /** Get a submatrix. 310 @param i0 Initial row index 311 @param i1 Final row index 312 @param j0 Initial column index 313 @param j1 Final column index 314 @return A(i0:i1,j0:j1) 315 @exception ArrayIndexOutOfBoundsException Submatrix indices 316 */ 317 318 public Matrix getMatrix (int i0, int i1, int j0, int j1) { 319 Matrix X = new Matrix(i1-i0+1,j1-j0+1); 320 double[][] B = X.getArray(); 321 try { 322 for (int i = i0; i <= i1; i++) { 323 for (int j = j0; j <= j1; j++) { 324 B[i-i0][j-j0] = A[i][j]; 325 } 326 } 327 } catch(ArrayIndexOutOfBoundsException e) { 328 throw new ArrayIndexOutOfBoundsException("Submatrix indices"); 329 } 330 return X; 331 } 332 333 /** Get a submatrix. 334 @param r Array of row indices. 335 @param c Array of column indices. 336 @return A(r(:),c(:)) 337 @exception ArrayIndexOutOfBoundsException Submatrix indices 338 */ 339 340 public Matrix getMatrix (int[] r, int[] c) { 341 Matrix X = new Matrix(r.length,c.length); 342 double[][] B = X.getArray(); 343 try { 344 for (int i = 0; i < r.length; i++) { 345 for (int j = 0; j < c.length; j++) { 346 B[i][j] = A[r[i]][c[j]]; 347 } 348 } 349 } catch(ArrayIndexOutOfBoundsException e) { 350 throw new ArrayIndexOutOfBoundsException("Submatrix indices"); 351 } 352 return X; 353 } 354 355 /** Get a submatrix. 356 @param i0 Initial row index 357 @param i1 Final row index 358 @param c Array of column indices. 359 @return A(i0:i1,c(:)) 360 @exception ArrayIndexOutOfBoundsException Submatrix indices 361 */ 362 363 public Matrix getMatrix (int i0, int i1, int[] c) { 364 Matrix X = new Matrix(i1-i0+1,c.length); 365 double[][] B = X.getArray(); 366 try { 367 for (int i = i0; i <= i1; i++) { 368 for (int j = 0; j < c.length; j++) { 369 B[i-i0][j] = A[i][c[j]]; 370 } 371 } 372 } catch(ArrayIndexOutOfBoundsException e) { 373 throw new ArrayIndexOutOfBoundsException("Submatrix indices"); 374 } 375 return X; 376 } 377 378 /** Get a submatrix. 379 @param r Array of row indices. 380 @param j0 Initial column index 381 @param j1 Final column index 382 @return A(r(:),j0:j1) 383 @exception ArrayIndexOutOfBoundsException Submatrix indices 384 */ 385 386 public Matrix getMatrix (int[] r, int j0, int j1) { 387 Matrix X = new Matrix(r.length,j1-j0+1); 388 double[][] B = X.getArray(); 389 try { 390 for (int i = 0; i < r.length; i++) { 391 for (int j = j0; j <= j1; j++) { 392 B[i][j-j0] = A[r[i]][j]; 393 } 394 } 395 } catch(ArrayIndexOutOfBoundsException e) { 396 throw new ArrayIndexOutOfBoundsException("Submatrix indices"); 397 } 398 return X; 399 } 400 401 /** Set a single element. 402 @param i Row index. 403 @param j Column index. 404 @param s A(i,j). 405 @exception ArrayIndexOutOfBoundsException 406 */ 407 408 public void set (int i, int j, double s) { 409 A[i][j] = s; 410 } 411 412 /** Set a submatrix. 413 @param i0 Initial row index 414 @param i1 Final row index 415 @param j0 Initial column index 416 @param j1 Final column index 417 @param X A(i0:i1,j0:j1) 418 @exception ArrayIndexOutOfBoundsException Submatrix indices 419 */ 420 421 public void setMatrix (int i0, int i1, int j0, int j1, Matrix X) { 422 try { 423 for (int i = i0; i <= i1; i++) { 424 for (int j = j0; j <= j1; j++) { 425 A[i][j] = X.get(i-i0,j-j0); 426 } 427 } 428 } catch(ArrayIndexOutOfBoundsException e) { 429 throw new ArrayIndexOutOfBoundsException("Submatrix indices"); 430 } 431 } 432 433 /** Set a submatrix. 434 @param r Array of row indices. 435 @param c Array of column indices. 436 @param X A(r(:),c(:)) 437 @exception ArrayIndexOutOfBoundsException Submatrix indices 438 */ 439 440 public void setMatrix (int[] r, int[] c, Matrix X) { 441 try { 442 for (int i = 0; i < r.length; i++) { 443 for (int j = 0; j < c.length; j++) { 444 A[r[i]][c[j]] = X.get(i,j); 445 } 446 } 447 } catch(ArrayIndexOutOfBoundsException e) { 448 throw new ArrayIndexOutOfBoundsException("Submatrix indices"); 449 } 450 } 451 452 /** Set a submatrix. 453 @param r Array of row indices. 454 @param j0 Initial column index 455 @param j1 Final column index 456 @param X A(r(:),j0:j1) 457 @exception ArrayIndexOutOfBoundsException Submatrix indices 458 */ 459 460 public void setMatrix (int[] r, int j0, int j1, Matrix X) { 461 try { 462 for (int i = 0; i < r.length; i++) { 463 for (int j = j0; j <= j1; j++) { 464 A[r[i]][j] = X.get(i,j-j0); 465 } 466 } 467 } catch(ArrayIndexOutOfBoundsException e) { 468 throw new ArrayIndexOutOfBoundsException("Submatrix indices"); 469 } 470 } 471 472 /** Set a submatrix. 473 @param i0 Initial row index 474 @param i1 Final row index 475 @param c Array of column indices. 476 @param X A(i0:i1,c(:)) 477 @exception ArrayIndexOutOfBoundsException Submatrix indices 478 */ 479 480 public void setMatrix (int i0, int i1, int[] c, Matrix X) { 481 try { 482 for (int i = i0; i <= i1; i++) { 483 for (int j = 0; j < c.length; j++) { 484 A[i][c[j]] = X.get(i-i0,j); 485 } 486 } 487 } catch(ArrayIndexOutOfBoundsException e) { 488 throw new ArrayIndexOutOfBoundsException("Submatrix indices"); 489 } 490 } 491 492 /** Matrix transpose. 493 @return A' 494 */ 495 496 public Matrix transpose () { 497 Matrix X = new Matrix(n,m); 498 double[][] C = X.getArray(); 499 for (int i = 0; i < m; i++) { 500 for (int j = 0; j < n; j++) { 501 C[j][i] = A[i][j]; 502 } 503 } 504 return X; 505 } 506 507 /** One norm 508 @return maximum column sum. 509 */ 510 511 public double norm1 () { 512 double f = 0; 513 for (int j = 0; j < n; j++) { 514 double s = 0; 515 for (int i = 0; i < m; i++) { 516 s += Math.abs(A[i][j]); 517 } 518 f = Math.max(f,s); 519 } 520 return f; 521 } 522 523 /** Two norm 524 @return maximum singular value. 525 */ 526 527 public double norm2 () { 528 return (new SingularValueDecomposition(this).norm2()); 529 } 530 531 /** Infinity norm 532 @return maximum row sum. 533 */ 534 535 public double normInf () { 536 double f = 0; 537 for (int i = 0; i < m; i++) { 538 double s = 0; 539 for (int j = 0; j < n; j++) { 540 s += Math.abs(A[i][j]); 541 } 542 f = Math.max(f,s); 543 } 544 return f; 545 } 546 547 /** Frobenius norm 548 @return sqrt of sum of squares of all elements. 549 */ 550 551 public double normF () { 552 double f = 0; 553 for (int i = 0; i < m; i++) { 554 for (int j = 0; j < n; j++) { 555 f = Maths.hypot(f,A[i][j]); 556 } 557 } 558 return f; 559 } 560 561 /** Unary minus 562 @return -A 563 */ 564 565 public Matrix uminus () { 566 Matrix X = new Matrix(m,n); 567 double[][] C = X.getArray(); 568 for (int i = 0; i < m; i++) { 569 for (int j = 0; j < n; j++) { 570 C[i][j] = -A[i][j]; 571 } 572 } 573 return X; 574 } 575 576 /** C = A + B 577 @param B another matrix 578 @return A + B 579 */ 580 581 public Matrix plus (Matrix B) { 582 checkMatrixDimensions(B); 583 Matrix X = new Matrix(m,n); 584 double[][] C = X.getArray(); 585 for (int i = 0; i < m; i++) { 586 for (int j = 0; j < n; j++) { 587 C[i][j] = A[i][j] + B.A[i][j]; 588 } 589 } 590 return X; 591 } 592 593 /** A = A + B 594 @param B another matrix 595 @return A + B 596 */ 597 598 public Matrix plusEquals (Matrix B) { 599 checkMatrixDimensions(B); 600 for (int i = 0; i < m; i++) { 601 for (int j = 0; j < n; j++) { 602 A[i][j] = A[i][j] + B.A[i][j]; 603 } 604 } 605 return this; 606 } 607 608 /** C = A - B 609 @param B another matrix 610 @return A - B 611 */ 612 613 public Matrix minus (Matrix B) { 614 checkMatrixDimensions(B); 615 Matrix X = new Matrix(m,n); 616 double[][] C = X.getArray(); 617 for (int i = 0; i < m; i++) { 618 for (int j = 0; j < n; j++) { 619 C[i][j] = A[i][j] - B.A[i][j]; 620 } 621 } 622 return X; 623 } 624 625 /** A = A - B 626 @param B another matrix 627 @return A - B 628 */ 629 630 public Matrix minusEquals (Matrix B) { 631 checkMatrixDimensions(B); 632 for (int i = 0; i < m; i++) { 633 for (int j = 0; j < n; j++) { 634 A[i][j] = A[i][j] - B.A[i][j]; 635 } 636 } 637 return this; 638 } 639 640 /** Element-by-element multiplication, C = A.*B 641 @param B another matrix 642 @return A.*B 643 */ 644 645 public Matrix arrayTimes (Matrix B) { 646 checkMatrixDimensions(B); 647 Matrix X = new Matrix(m,n); 648 double[][] C = X.getArray(); 649 for (int i = 0; i < m; i++) { 650 for (int j = 0; j < n; j++) { 651 C[i][j] = A[i][j] * B.A[i][j]; 652 } 653 } 654 return X; 655 } 656 657 /** Element-by-element multiplication in place, A = A.*B 658 @param B another matrix 659 @return A.*B 660 */ 661 662 public Matrix arrayTimesEquals (Matrix B) { 663 checkMatrixDimensions(B); 664 for (int i = 0; i < m; i++) { 665 for (int j = 0; j < n; j++) { 666 A[i][j] = A[i][j] * B.A[i][j]; 667 } 668 } 669 return this; 670 } 671 672 /** Element-by-element right division, C = A./B 673 @param B another matrix 674 @return A./B 675 */ 676 677 public Matrix arrayRightDivide (Matrix B) { 678 checkMatrixDimensions(B); 679 Matrix X = new Matrix(m,n); 680 double[][] C = X.getArray(); 681 for (int i = 0; i < m; i++) { 682 for (int j = 0; j < n; j++) { 683 C[i][j] = A[i][j] / B.A[i][j]; 684 } 685 } 686 return X; 687 } 688 689 /** Element-by-element right division in place, A = A./B 690 @param B another matrix 691 @return A./B 692 */ 693 694 public Matrix arrayRightDivideEquals (Matrix B) { 695 checkMatrixDimensions(B); 696 for (int i = 0; i < m; i++) { 697 for (int j = 0; j < n; j++) { 698 A[i][j] = A[i][j] / B.A[i][j]; 699 } 700 } 701 return this; 702 } 703 704 /** Element-by-element left division, C = A.\B 705 @param B another matrix 706 @return A.\B 707 */ 708 709 public Matrix arrayLeftDivide (Matrix B) { 710 checkMatrixDimensions(B); 711 Matrix X = new Matrix(m,n); 712 double[][] C = X.getArray(); 713 for (int i = 0; i < m; i++) { 714 for (int j = 0; j < n; j++) { 715 C[i][j] = B.A[i][j] / A[i][j]; 716 } 717 } 718 return X; 719 } 720 721 /** Element-by-element left division in place, A = A.\B 722 @param B another matrix 723 @return A.\B 724 */ 725 726 public Matrix arrayLeftDivideEquals (Matrix B) { 727 checkMatrixDimensions(B); 728 for (int i = 0; i < m; i++) { 729 for (int j = 0; j < n; j++) { 730 A[i][j] = B.A[i][j] / A[i][j]; 731 } 732 } 733 return this; 734 } 735 736 /** Multiply a matrix by a scalar, C = s*A 737 @param s scalar 738 @return s*A 739 */ 740 741 public Matrix times (double s) { 742 Matrix X = new Matrix(m,n); 743 double[][] C = X.getArray(); 744 for (int i = 0; i < m; i++) { 745 for (int j = 0; j < n; j++) { 746 C[i][j] = s*A[i][j]; 747 } 748 } 749 return X; 750 } 751 752 /** Multiply a matrix by a scalar in place, A = s*A 753 @param s scalar 754 @return replace A by s*A 755 */ 756 757 public Matrix timesEquals (double s) { 758 for (int i = 0; i < m; i++) { 759 for (int j = 0; j < n; j++) { 760 A[i][j] = s*A[i][j]; 761 } 762 } 763 return this; 764 } 765 766 /** Linear algebraic matrix multiplication, A * B 767 @param B another matrix 768 @return Matrix product, A * B 769 @exception IllegalArgumentException Matrix inner dimensions must agree. 770 */ 771 772 public Matrix times (Matrix B) { 773 if (B.m != n) { 774 throw new IllegalArgumentException("Matrix inner dimensions must agree."); 775 } 776 Matrix X = new Matrix(m,B.n); 777 double[][] C = X.getArray(); 778 double[] Bcolj = new double[n]; 779 for (int j = 0; j < B.n; j++) { 780 for (int k = 0; k < n; k++) { 781 Bcolj[k] = B.A[k][j]; 782 } 783 for (int i = 0; i < m; i++) { 784 double[] Arowi = A[i]; 785 double s = 0; 786 for (int k = 0; k < n; k++) { 787 s += Arowi[k]*Bcolj[k]; 788 } 789 C[i][j] = s; 790 } 791 } 792 return X; 793 } 794 795 /** LU Decomposition 796 @return LUDecomposition 797 @see LUDecomposition 798 */ 799 800 public LUDecomposition lu () { 801 return new LUDecomposition(this); 802 } 803 804 /** QR Decomposition 805 @return QRDecomposition 806 @see QRDecomposition 807 */ 808 809 public QRDecomposition qr () { 810 return new QRDecomposition(this); 811 } 812 813 /** Cholesky Decomposition 814 @return CholeskyDecomposition 815 @see CholeskyDecomposition 816 */ 817 818 public CholeskyDecomposition chol () { 819 return new CholeskyDecomposition(this); 820 } 821 822 /** Singular Value Decomposition 823 @return SingularValueDecomposition 824 @see SingularValueDecomposition 825 */ 826 827 public SingularValueDecomposition svd () { 828 return new SingularValueDecomposition(this); 829 } 830 831 /** Eigenvalue Decomposition 832 @return EigenvalueDecomposition 833 @see EigenvalueDecomposition 834 */ 835 836 public EigenvalueDecomposition eig () { 837 return new EigenvalueDecomposition(this); 838 } 839 840 /** Solve A*X = B 841 @param B right hand side 842 @return solution if A is square, least squares solution otherwise 843 */ 844 845 public Matrix solve (Matrix B) { 846 return (m == n ? (new LUDecomposition(this)).solve(B) : 847 (new QRDecomposition(this)).solve(B)); 848 } 849 850 /** Solve X*A = B, which is also A'*X' = B' 851 @param B right hand side 852 @return solution if A is square, least squares solution otherwise. 853 */ 854 855 public Matrix solveTranspose (Matrix B) { 856 return transpose().solve(B.transpose()); 857 } 858 859 /** Matrix inverse or pseudoinverse 860 @return inverse(A) if A is square, pseudoinverse otherwise. 861 */ 862 863 public Matrix inverse () { 864 return solve(identity(m,m)); 865 } 866 867 /** Matrix determinant 868 @return determinant 869 */ 870 871 public double det () { 872 return new LUDecomposition(this).det(); 873 } 874 875 /** Matrix rank 876 @return effective numerical rank, obtained from SVD. 877 */ 878 879 public int rank () { 880 return new SingularValueDecomposition(this).rank(); 881 } 882 883 /** Matrix condition (2 norm) 884 @return ratio of largest to smallest singular value. 885 */ 886 887 public double cond () { 888 return new SingularValueDecomposition(this).cond(); 889 } 890 891 /** Matrix trace. 892 @return sum of the diagonal elements. 893 */ 894 895 public double trace () { 896 double t = 0; 897 for (int i = 0; i < Math.min(m,n); i++) { 898 t += A[i][i]; 899 } 900 return t; 901 } 902 903 /** Generate matrix with random elements 904 @param m Number of rows. 905 @param n Number of colums. 906 @return An m-by-n matrix with uniformly distributed random elements. 907 */ 908 909 public static Matrix random (int m, int n) { 910 Matrix A = new Matrix(m,n); 911 double[][] X = A.getArray(); 912 for (int i = 0; i < m; i++) { 913 for (int j = 0; j < n; j++) { 914 X[i][j] = Math.random(); 915 } 916 } 917 return A; 918 } 919 920 /** Generate identity matrix 921 @param m Number of rows. 922 @param n Number of colums. 923 @return An m-by-n matrix with ones on the diagonal and zeros elsewhere. 924 */ 925 926 public static Matrix identity (int m, int n) { 927 Matrix A = new Matrix(m,n); 928 double[][] X = A.getArray(); 929 for (int i = 0; i < m; i++) { 930 for (int j = 0; j < n; j++) { 931 X[i][j] = (i == j ? 1.0 : 0.0); 932 } 933 } 934 return A; 935 } 936 937 @Override 938public String toString(){ 939 StringWriter writer = new StringWriter(); 940 PrintWriter printWriter = new PrintWriter(writer); 941 print(printWriter,getColumnDimension(),3); 942 return writer.toString(); 943 } 944 945 946 /** Print the matrix to stdout. Line the elements up in columns 947 * with a Fortran-like 'Fw.d' style format. 948 @param w Column width. 949 @param d Number of digits after the decimal. 950 */ 951 952 public void print (int w, int d) { 953 print(new PrintWriter(System.out,true),w,d); } 954 955 /** Print the matrix to the output stream. Line the elements up in 956 * columns with a Fortran-like 'Fw.d' style format. 957 @param output Output stream. 958 @param w Column width. 959 @param d Number of digits after the decimal. 960 */ 961 962 public void print (PrintWriter output, int w, int d) { 963 DecimalFormat format = new DecimalFormat(); 964 format.setDecimalFormatSymbols(new DecimalFormatSymbols(Locale.US)); 965 format.setMinimumIntegerDigits(1); 966 format.setMaximumFractionDigits(d); 967 format.setMinimumFractionDigits(d); 968 format.setGroupingUsed(false); 969 print(output,format,w+2); 970 } 971 972 /** Print the matrix to stdout. Line the elements up in columns. 973 * Use the format object, and right justify within columns of width 974 * characters. 975 * Note that is the matrix is to be read back in, you probably will want 976 * to use a NumberFormat that is set to US Locale. 977 @param format A Formatting object for individual elements. 978 @param width Field width for each column. 979 @see java.text.DecimalFormat#setDecimalFormatSymbols 980 */ 981 982 public void print (NumberFormat format, int width) { 983 print(new PrintWriter(System.out,true),format,width); } 984 985 // DecimalFormat is a little disappointing coming from Fortran or C's printf. 986 // Since it doesn't pad on the left, the elements will come out different 987 // widths. Consequently, we'll pass the desired column width in as an 988 // argument and do the extra padding ourselves. 989 990 /** Print the matrix to the output stream. Line the elements up in columns. 991 * Use the format object, and right justify within columns of width 992 * characters. 993 * Note that is the matrix is to be read back in, you probably will want 994 * to use a NumberFormat that is set to US Locale. 995 @param output the output stream. 996 @param format A formatting object to format the matrix elements 997 @param width Column width. 998 @see java.text.DecimalFormat#setDecimalFormatSymbols 999 */ 1000 1001 public void print (PrintWriter output, NumberFormat format, int width) { 1002 output.println(); // start on new line. 1003 for (int i = 0; i < m; i++) { 1004 for (int j = 0; j < n; j++) { 1005 String s = format.format(A[i][j]); // format the number 1006 int padding = Math.max(1,width-s.length()); // At _least_ 1 space 1007 for (int k = 0; k < padding; k++) 1008 output.print(' '); 1009 output.print(s); 1010 } 1011 output.println(); 1012 } 1013 //output.println(); // end with blank line. 1014 } 1015 1016 /** Read a matrix from a stream. The format is the same the print method, 1017 * so printed matrices can be read back in (provided they were printed using 1018 * US Locale). Elements are separated by 1019 * whitespace, all the elements for each row appear on a single line, 1020 * the last row is followed by a blank line. 1021 @param input the input stream. 1022 @return a Matrix 1023 @throws java.io.IOException 1024 */ 1025 1026 @SuppressWarnings({ "rawtypes", "unchecked" }) 1027public static Matrix read (BufferedReader input) throws java.io.IOException { 1028 StreamTokenizer tokenizer= new StreamTokenizer(input); 1029 1030 // Although StreamTokenizer will parse numbers, it doesn't recognize 1031 // scientific notation (E or D); however, Double.valueOf does. 1032 // The strategy here is to disable StreamTokenizer's number parsing. 1033 // We'll only get whitespace delimited words, EOL's and EOF's. 1034 // These words should all be numbers, for Double.valueOf to parse. 1035 1036 tokenizer.resetSyntax(); 1037 tokenizer.wordChars(0,255); 1038 tokenizer.whitespaceChars(0, ' '); 1039 tokenizer.eolIsSignificant(true); 1040 java.util.Vector v = new java.util.Vector(); 1041 1042 // Ignore initial empty lines 1043 while (tokenizer.nextToken() == StreamTokenizer.TT_EOL); 1044 if (tokenizer.ttype == StreamTokenizer.TT_EOF) 1045 throw new java.io.IOException("Unexpected EOF on matrix read."); 1046 do { 1047 v.addElement(Double.valueOf(tokenizer.sval)); // Read & store 1st row. 1048 } while (tokenizer.nextToken() == StreamTokenizer.TT_WORD); 1049 1050 int n = v.size(); // Now we've got the number of columns! 1051 double[] row = new double[n]; 1052 for (int j=0; j<n; j++) // extract the elements of the 1st row. 1053 row[j]=((Double)v.elementAt(j)).doubleValue(); 1054 v.removeAllElements(); 1055 v.addElement(row); // Start storing rows instead of columns. 1056 while (tokenizer.nextToken() == StreamTokenizer.TT_WORD) { 1057 // While non-empty lines 1058 v.addElement(row = new double[n]); 1059 int j = 0; 1060 do { 1061 if (j >= n) throw new java.io.IOException 1062 ("Row " + v.size() + " is too long."); 1063 row[j++] = Double.parseDouble(tokenizer.sval); 1064 } while (tokenizer.nextToken() == StreamTokenizer.TT_WORD); 1065 if (j < n) throw new java.io.IOException 1066 ("Row " + v.size() + " is too short."); 1067 } 1068 int m = v.size(); // Now we've got the number of rows. 1069 double[][] A = new double[m][]; 1070 v.copyInto(A); // copy the rows out of the vector 1071 return new Matrix(A); 1072 } 1073 1074 1075/* ------------------------ 1076 Private Methods 1077 * ------------------------ */ 1078 1079 /** Check if size(A) == size(B) **/ 1080 1081 private void checkMatrixDimensions (Matrix B) { 1082 if (B.m != m || B.n != n) { 1083 throw new IllegalArgumentException("Matrix dimensions must agree."); 1084 } 1085 } 1086 1087}