如何将块转换为块对角矩阵(NumPy)

17 投票
4 回答
9124 浏览
提问于 2025-04-16 06:54

我有三个大小相同的方阵,都是用NumPy这个库做的。我想把这三个方阵合并成一个块对角矩阵。

举个例子:

a1 = np.array([[1,1,1],[1,1,1],[1,1,1]])
a2 = np.array([[2,2,2],[2,2,2],[2,2,2]])
a3 = np.array([[3,3,3],[3,3,3],[3,3,3]])

r = np.array([[1,1,1,0,0,0,0,0,0],[1,1,1,0,0,0,0,0,0],[1,1,1,0,0,0,0,0,0],[0,0,0,2,2,2,0,0,0],[0,0,0,2,2,2,0,0,0],[0,0,0,2,2,2,0,0,0],[0,0,0,0,0,0,3,3,3],[0,0,0,0,0,0,3,3,3],[0,0,0,0,0,0,3,3,3]])

请问,最好的方法是什么呢?

4 个回答

3

如果你想要这个特定的数组 r,也许最简单的方法就是:

r=np.kron(np.diag([1,2,3]),np.ones((3,3),dtype='int'))

如果 a1a2a3 可以是任意的二维数组,那么最简单的方法可能是:

a1=np.asmatrix(a1)
a2=np.asmatrix(a2)
a3=np.asmatrix(a3)
z=np.asmatrix(np.zeros((3,3)))
r=np.asarray(np.bmat('a1, z, z; z, a2, z; z, z, a3'))

还有一种比较慢的方法是:

r=(np.kron([[1,0,0],[0,0,0],[0,0,0]],a1)   
   +np.kron([[0,0,0],[0,1,0],[0,0,0]],a2)
   +np.kron([[0,0,0],[0,0,0],[0,0,1]],a3))
4

自从这些回答出现以来,numpy 增加了一个叫做 block 的功能。

In [695]: A=np.arange(1,10).reshape(3,3)
In [696]: B=np.arange(10,14).reshape(2,2)
In [698]: C = np.zeros((3,2),int)

In [699]: np.block([[A,C],[C.T,B]])
Out[699]: 
array([[ 1,  2,  3,  0,  0],
       [ 4,  5,  6,  0,  0],
       [ 7,  8,  9,  0,  0],
       [ 0,  0,  0, 10, 11],
       [ 0,  0,  0, 12, 13]])
22

scipy.linalg 里有一个叫做 block_diag 的函数,可以自动完成这个操作。

>>> a1 = np.array([[1,1,1],[1,1,1],[1,1,1]])
>>> a2 = np.array([[2,2,2],[2,2,2],[2,2,2]])
>>> a3 = np.array([[3,3,3],[3,3,3],[3,3,3]])
>>> import scipy.linalg
>>> scipy.linalg.block_diag(a1, a2, a3)
array([[1, 1, 1, 0, 0, 0, 0, 0, 0],
       [1, 1, 1, 0, 0, 0, 0, 0, 0],
       [1, 1, 1, 0, 0, 0, 0, 0, 0],
       [0, 0, 0, 2, 2, 2, 0, 0, 0],
       [0, 0, 0, 2, 2, 2, 0, 0, 0],
       [0, 0, 0, 2, 2, 2, 0, 0, 0],
       [0, 0, 0, 0, 0, 0, 3, 3, 3],
       [0, 0, 0, 0, 0, 0, 3, 3, 3],
       [0, 0, 0, 0, 0, 0, 3, 3, 3]])
>>> r = np.array([[1,1,1,0,0,0,0,0,0],[1,1,1,0,0,0,0,0,0],[1,1,1,0,0,0,0,0,0], [0,0,0,2,2,2,0,0,0],[0,0,0,2,2,2,0,0,0],[0,0,0,2,2,2,0,0,0],[0,0,0,0,0,0,3,3,3],[0,0,0,0,0,0,3,3,3],[0,0,0,0,0,0,3,3,3]])
>>> (scipy.linalg.block_diag(a1, a2, a3)  == r).all()
True

撰写回答