我有一个numpy数组,其中包含8个16字节长的空白记录,如下所示:
array([[0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0],
[0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0],
[0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0],
[0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0],
[0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0],
[0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0],
[0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0],
[0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0]],
dtype='|V16')
并且需要将其转换为具有8个16B长记录的自定义dtype的数组,如下所示:
[(('x', 's0'), '<u4'), (('y', 's1'), '<u4'), (('z', 's2'), '<u4'), ('padding0', '<u4')]
我怎样才能做到这一点?
我试过array.astype(self.dtype, copy=False, casting="unsafe")
,
但是我得到了
ValueError:使用序列设置数组元素。
这对我来说没什么意义。
这个数据来自PyOpenCL(内存映射缓冲区),我无法真正改变输入格式或dtype。
答案 0 :(得分:1)
只要字节数匹配,view
就可以进行这种转换。它只是改变了“查看”数据缓冲区的方式。
In [36]: dt=np.dtype([(('x', 's0'), '<u4'), (('y', 's1'), '<u4'), (('z', 's2'), '<u4'), ('padding0', '<u4')])
In [37]: dt
Out[37]: dtype([(('x', 's0'), '<u4'), (('y', 's1'), '<u4'), (('z', 's2'), '<u4'), ('padding0', '<u4')])
In [39]: x = np.zeros((3,), dtype=dt)
In [40]: x
Out[40]:
array([(0, 0, 0, 0), (0, 0, 0, 0), (0, 0, 0, 0)],
dtype=[(('x', 's0'), '<u4'), (('y', 's1'), '<u4'), (('z', 's2'), '<u4'), ('padding0', '<u4')])
In [41]: x.view('|V16')
Out[41]:
array([[0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0],
[0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0], [0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0]],
dtype='|V16')
In [42]: x.view('|V16').view(dt)
Out[42]:
array([(0, 0, 0, 0), (0, 0, 0, 0), (0, 0, 0, 0)],
dtype=[(('x', 's0'), '<u4'), (('y', 's1'), '<u4'), (('z', 's2'), '<u4'), ('padding0', '<u4')])
我经常需要尝试确定astype
或view
是否是转换结构化数组的正确方法。