「.NET 開発基盤部会 Wiki」は、「Open棟梁Project」,「OSSコンソーシアム .NET開発基盤部会」によって運営されています。
多次元配列における数値計算を効率的に行うことをサポートする数学関数ライブラリ
> pip install numpy
>>>import numpy as np
1次元配列はベクトルとも言う。
>>>x=np.array([1.0,2.0,3.0]) >>> x array([1., 2., 3.]) >>>print(x) [1. 2. 3.] >>>type(x) <class 'numpy.ndarray'>
>>>np.array([[1, 2], [3, 4]]) array([[1, 2], [3, 4]])
>>>np.arange(10) array([0, 1, 2, 3, 4, 5, 6, 7, 8, 9])
>>>np.ones(10) array([1., 1., 1., 1., 1., 1., 1., 1., 1., 1.])
>>>np.zeros(10) array([0., 0., 0., 0., 0., 0., 0., 0., 0., 0.])
>>>np.empty(5) array([2.12199579e-314, 6.36598737e-314, 1.06099790e-313, 1.48539705e-313, 1.90979621e-313])
>>>np.random.randn(10) array([ 1.33385073, -0.61769365, 1.41045502, -0.18429257, -0.00662791, -0.09483573, -2.26232104, 0.95660959, -1.02715574, -1.02119134])
>>>x=np.array([1.0,2.0,3.0]) >>> np.ndim(x) # 配列の次元 1 >>> x.shape # 次元の形状 (3,) >>> x.shape[0] # 次元の要素数 3
同じ数の要素を持つ2つのNumPy配列を算術計算で処理する。
>>>x=np.array([1.0,2.0,3.0]) >>>y=np.array([2.0,4.0,6.0])
>>>x+y array([ 3., 6., 9.])
>>>x-y array([-1., -2., -3.])
>>>x*y array([ 2., 8., 18.])
>>>x/y array([ 0.5, 0.5, 0.5])
arr = np.arange(10)
arr
>>>np.sqrt(arr) array([0. , 1. , 1.41421356, 1.73205081, 2. , 2.23606798, 2.44948974, 2.64575131, 2.82842712, 3. ])
>>>np.exp(arr) array([1.00000000e+00, 2.71828183e+00, 7.38905610e+00, 2.00855369e+01, 5.45981500e+01, 1.48413159e+02, 4.03428793e+02, 1.09663316e+03, 2.98095799e+03, 8.10308393e+03])
>>>x=np.random.randn(10)
>>>np.sign(x) array([ 1., 1., -1., -1., -1., -1., 1., 1., 1., 1.])
>>>x.mean() 0.02713878708811851
>>>x.std() 1.2305510107988773
行列とは?
列↓ ┌ ┐ 行│○ ○│ →│ │ │○ ○│ └ ┘
テンソルとは?
>>> a=np.array([[1,2],[3,4],[5,6]]) >>> print(a) [[1 2] [3 4] [5 6]] >>> np.ndim(a) # 配列の次元 2 >>> a.shape # 行列の形状 (3, 2) # 行数・列数 >>> a.dtype # 行列要素のデータ型 dtype('int32')
>>>a=np.array([[1, 2], [3, 4]]) >>>b=np.array([[3, 0], [0, 6]]) >>>a+b array([[ 4, 2], [ 3, 10]])
>>>a=np.array([[1, 2], [3, 4]]) >>>b=np.array([10,20]) >>>a+b array([[11, 22], [13, 24]])
>>>a=np.array([[1, 2], [3, 4]]) >>>b=np.array([[3, 0], [0, 6]]) >>>a*b array([[ 3, 0], [ 0, 24]])
>>>a=np.array([[1, 2], [3, 4]]) >>>b=np.array([10,20]) >>>a*b array([[10, 40], [30, 80]])
>>> a=np.array([[51, 55], [14, 19], [0, 4]]) >>> print(a) [[51 55] [14 19] [ 0 4]]
>>>a[0][1] 55
>>>a[0] array([51, 55])
>>>a[0,1:2] array([55]) >>>a[1:2,0] array([14])
>>>a[np.array([0,1])] array([[51, 55], [14, 19]])
>>>a>15 array([[ True, True], [False, True], [False, False]])
>>>a[a>15] array([51, 55, 19])
>>> a=a.flatten() >>> print(a) [51 55 14 19 0 4]
np[0,:] # 1行目、全列スライス
np[:,0] # 1列目、全行スライス
>>>np[0,:].reshape(1, -1)
>>>np[:,0].reshape(1, -1)
スカラ値や異なる形状の配列と計算を行う。
配列とスカラ値との計算では、スカラ値が配列形状に拡張され計算される。
>>>x=np.array([1.0,2.0,3.0]) >>>x/2.0 array([0.5, 1. , 1.5])
>>>a=np.array([[1, 2], [3, 4]]) >>>a*10 array([[10, 20], [30, 40]])
→ 行列の積算
>>> a=np.array([[51, 55], [14, 19], [0, 4]]) >>> print(a) [[51 55] [14 19] [ 0 4]]
>>>print(a.T) [[51 14 0] [55 19 4]]
>>>print(a.transpose()) [[51 14 0] [55 19 4]]
A行列 * B行列 =C行列
┌ ┐┌ ┐ ┌ ┐ │a1 b1││a2 b2│ │a1a2+b1c2 a1b2+b1d2│ │ ││ │ = │ │ │c1 d1││c2 d2│ │c1a2+d1c2 c1b2+d1d2│ └ ┘└ ┘ └ ┘ A行列 B行列 C行列 2行2列 2行2列 2行2列
計算後のC行列は、A行列の行数 * B行列の列数の行列になる。
┌ ┐┌ ┐ ┌ ┐┌ ┐ ┌ ┐ │a1 b1││a2│ │a1 b1││a2 0│ │a1a2+b1b2│ │ ││ │ = │ ││ │ =│ │ │c1 d1││b2│ │c1 d1││b2 0│ │c1a2+d1b2│ └ ┘└ ┘ └ ┘└ ┘ └ ┘ A行列 B行列 C行列 2行2列 2行1列 2行1列
┌ ┐┌ ┐ ┌ ┐ │a1 b1││a2 b2 c2 d2│ │a1a2+b1e2 a1b2+b1f2 a1c2+b1g2 a1d2+b1h2│ │ ││ │ │ │ │c1 d1││e2 f2 g2 h2│ = │c1a2+d1e2 c1b2+d1f2 c1c2+d1g2 c1d2+d1h2│ │ │└ ┘ │ │ │e1 f1│ │e1a2+f1e2 e1b2+f1f2 e1c2+f1g2 e1d2+f1h2│ └ ┘ └ ┘ A行列 B行列 C行列 3行2列 2行4列 3行4列
要素ごとの掛け算でないことに注意
>>>a=np.array([[1,2,3],[4,5,6]]) >>>b=np.array([[1,2],[3,4],[5,6]]) >>>c=np.array([[1,2],[3,4]]) >>>d=np.array([7,8]) >>>a array([[1, 2, 3], [4, 5, 6]]) >>>b array([[1, 2], [3, 4], [5, 6]]) >>>c array([[1, 2], [3, 4]]) >>>d array([7, 8])
>>>a@b array([[22, 28], [49, 64]]) >>>b@c array([[ 7, 10], [15, 22], [23, 34]]) >>>b@d array([23, 53, 83]) >>>a@c # A行列の列数とB行列の行数が一致しない。 Traceback (most recent call last): File "<stdin>", line 1, in <module> ValueError: matmul: Input operand 1 has a mismatch in its core dimension 0, with gufunc signature (n?,k),(k,m?)->(n?,m?) (size 2 is different from 3)
>>>np.dot(a,b) array([[22, 28], [49, 64]]) >>>np.dot(b,c) array([[ 7, 10], [15, 22], [23, 34]]) >>>np.dot(b,d) array([23, 53, 83]) >>>np.dot(a,c) # A行列の列数とB行列の行数が一致しない。 Traceback (most recent call last): File "<stdin>", line 1, in <module> ValueError: shapes (2,3) and (2,2) not aligned: 3 (dim 1) != 2 (dim 0)
>>>a=np.array([1,2,3,4]) >>>b=np.array([10,20,30,40])
ベクトルを行列化する。
>>> a=a.reshape(1, -1) >>> b=b.reshape(1, -1).T # 転置行列に
>>>a@b
>>>np.dot(a,b)
行列では、交換法則の不成立になるが、
ベクトルの場合は成立する(ただし転置が必要)。
>>>a@b >>>b.T@a.T
>>>np.dot(a,b) >>>np.dot(b.T,a.T)