矩阵向量与python / numpy

时间:2016-09-11 21:03:35

标签: python numpy

如果我需要通过按行或按列读取来创建向量,那么Numpy ravel效果很好。但是,我想通过使用图像处理中经常使用的方法将矩阵转换为1d数组。这是一个包含初始矩阵A和最终结果B的示例:

A = np.array([[ 0,  1,  2,  3],
              [ 4,  5,  6,  7],
              [ 8,  9, 10, 11],
              [12, 13, 14, 15]])

B = np.array([[ 0, 1, 4, 8, 5, 2, 3, 6, 9, 12, 13, 10, 7, 11, 14, 15])

是否已有可以帮助我的现有功能?如果没有,你能给我一些关于如何解决这个问题的提示吗? PS。矩阵ANxN

2 个答案:

答案 0 :(得分:4)

我已经使用numpy好几年了,而且我从未见过这样的功能。

这是你可以做到的一种方式(不一定是效率最高的):

In [47]: a
Out[47]: 
array([[ 0,  1,  2,  3],
       [ 4,  5,  6,  7],
       [ 8,  9, 10, 11],
       [12, 13, 14, 15]])

In [48]: np.concatenate([np.diagonal(a[::-1,:], k)[::(2*(k % 2)-1)] for k in range(1-a.shape[0], a.shape[0])])
Out[48]: array([ 0,  1,  4,  8,  5,  2,  3,  6,  9, 12, 13, 10,  7, 11, 14, 15])

将单行分解为单独的步骤:

a[::-1, :]撤消行:

In [59]: a[::-1, :]
Out[59]: 
array([[12, 13, 14, 15],
       [ 8,  9, 10, 11],
       [ 4,  5,  6,  7],
       [ 0,  1,  2,  3]])

(也可以写成a[::-1]np.flipud(a)。)

np.diagonal(a, k)提取k对角线,其中k=0是主对角线。所以,例如,

In [65]: np.diagonal(a[::-1, :], -3)
Out[65]: array([0])

In [66]: np.diagonal(a[::-1, :], -2)
Out[66]: array([4, 1])

In [67]: np.diagonal(a[::-1, :], 0)
Out[67]: array([12,  9,  6,  3])

In [68]: np.diagonal(a[::-1, :], 2)
Out[68]: array([14, 11])

在列表理解中,k给出了要提取的对角线。我们想要反转每个其他对角线的元素。表达式2*(k % 2) - 1给出值1,-1,1,......,因为k在-3到3之间变化。使用[::1]进行索引会使索引的数组顺序保持不变,并使用[::-1]建立索引可以反转数组的顺序。所以np.diagonal(a[::-1, :], k)[::(2*(k % 2)-1)]给出了k对角线,但每个其他对角线都反转了:

In [71]: [np.diagonal(a[::-1,:], k)[::(2*(k % 2)-1)] for k in range(1-a.shape[0], a.shape[0])]
Out[71]: 
[array([0]),
 array([1, 4]),
 array([8, 5, 2]),
 array([ 3,  6,  9, 12]),
 array([13, 10,  7]),
 array([11, 14]),
 array([15])]

np.concatenate()将它们全部放入一个数组中:

In [72]: np.concatenate([np.diagonal(a[::-1,:], k)[::(2*(k % 2)-1)] for k in range(1-a.shape[0], a.shape[0])])
Out[72]: array([ 0,  1,  4,  8,  5,  2,  3,  6,  9, 12, 13, 10,  7, 11, 14, 15])

答案 1 :(得分:1)

我发现对于MATLAB的zigzag扫描的讨论,但对numpy没有多少讨论。一个项目似乎使用硬编码索引数组8x8块

https://github.com/lot9s/lfv-compression/blob/master/scripts/our_mpeg/zigzag.py

ZIG = np.array([[0,  1,  5,  6,  14, 15, 27, 28],
               [2,  4,  7,  13, 16, 26, 29, 42],
               [3,  8,  12, 17, 25, 30, 41, 43],
               [9,  11, 18, 24, 31, 40, 44,53],
               [10, 19, 23, 32, 39, 45, 52,54],
               [20, 22, 33, 38, 46, 51, 55,60],
               [21, 34, 37, 47, 50, 56, 59,61],
               [35, 36, 48, 49, 57, 58, 62,63]])

显然它使用了jpeg和mpeg压缩。