NArrayの次元が FORTRAN style なので、NMatrixの次元の順序は 数学表記の添字の順序とは逆になります。
各クラスのクラス次元は次のようになります。
NMatrix/NVector と NArray の演算では、NArray は Scalarとして機能します。 オブジェクトの持つ次元からクラス次元を引いた残りは NArrayの多次元配列と同じように機能します。 (サイズ1の次元の繰り返しなど。)
これにより、3次元 Vector の 100x100 の 2次元配列に Matrix を掛けて座標変換、というような場合でも、 スクリプトでループを書く必要がないので、非常に高速に処理できます。
例:
% irb -r nmatrix irb(main):001:0> m = NMatrix.float(2,2).indgen! NMatrix.float(2,2): [ [ 0.0, 1.0 ], [ 2.0, 3.0 ] ] irb(main):002:0> a = NArray.float(2).indgen!+1 NArray.float(2): [ 1.0, 2.0 ] irb(main):003:0> a * m NMatrix.float(2,2,2): [ [ [ 0.0, 1.0 ], [ 2.0, 3.0 ] ], [ [ 0.0, 2.0 ], [ 4.0, 6.0 ] ] ] irb(main):004:0> a + m TypeError: Illegal operation: NArray + NMatrix ./nmatrix.rb:109:in `coerce_rev' (irb):4:in `+' (irb):4:in `irb_binding' |
% irb -r nmatrix irb(main):001:0> m = NMatrix.float(2,2,2).indgen! NMatrix.float(2,2,2): [ [ [ 0.0, 1.0 ], [ 2.0, 3.0 ] ], [ [ 4.0, 5.0 ], [ 6.0, 7.0 ] ] ] irb(main):002:0> m[1,1,true] NArray.float(2): [ 3.0, 7.0 ] irb(main):003:0> m[0..1,1,true] NMatrix.float(2,1,2): [ [ [ 2.0, 3.0 ] ], [ [ 6.0, 7.0 ] ] ] |
y = m * x |
x = y / m |
x = m.lu.solve(y) |
また、
x = y / m |
x = m**-1 * y |
なお、LU分解は Ruby Object 型 配列でも使えます。
% irb -r nmatrix -r rational irb(main):001:0> m = NMatrix.object(4,4).collect!{Rational(rand(10))} NMatrix.object(4,4): [ [ Rational(3, 1), Rational(6, 1), Rational(5, 1), Rational(3, 1) ], [ Rational(1, 1), Rational(6, 1), Rational(2, 1), Rational(0, 1) ], [ Rational(0, 1), Rational(3, 1), Rational(1, 1), Rational(7, 1) ], [ Rational(2, 1), Rational(8, 1), Rational(9, 1), Rational(1, 1) ] ] irb(main):002:0> m/m NMatrix.object(4,4): [ [ Rational(1, 1), Rational(0, 1), Rational(0, 1), Rational(0, 1) ], [ Rational(0, 1), Rational(1, 1), Rational(0, 1), Rational(0, 1) ], [ Rational(0, 1), Rational(0, 1), Rational(1, 1), Rational(0, 1) ], [ Rational(0, 1), Rational(0, 1), Rational(0, 1), Rational(1, 1) ] ] |