uint8
65536の値を格納するルックアップテーブル(LUT)があります。
lut = np.random.randint(256, size=(65536,)).astype('uint8')
uint16
このLUTを使用して、 sの配列の値を変換したいと思います。
arr = np.random.randint(65536, size=(1000, 1000)).astype('uint16')
この最後の配列はかなり大きくなる可能性があるため、変換を適切に実行したいと思います。試してみると、次のようになります。
>>> np.take(lut, arr, out=arr)
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "C:\Python27\lib\site-packages\numpy\core\fromnumeric.py", line 103, in take
return take(indices, axis, out, mode)
TypeError: array cannot be safely cast to required type
そして、私は何が起こっているのか理解していません。引数がないout
場合、戻り値はと同じdtypeであることがわかっているlut
ので、uint8
。しかし、なぜuint8
キャストできないのuint16
ですか?あなたがnumpyに尋ねるなら:
>>> np.can_cast('uint8', 'uint16')
True
明らかに次の作品:
>>> lut = lut.astype('uint16')
>>> np.take(lut, arr, out=arr)
array([[173, 251, 218, ..., 110, 98, 235],
[200, 231, 91, ..., 158, 100, 88],
[ 13, 227, 223, ..., 94, 56, 36],
...,
[ 28, 198, 80, ..., 60, 87, 118],
[156, 46, 118, ..., 212, 198, 218],
[203, 97, 245, ..., 3, 191, 173]], dtype=uint16)
しかし、これも機能します。
>>> lut = lut.astype('int32')
>>> np.take(lut, arr, out=arr)
array([[ 78, 249, 148, ..., 77, 12, 167],
[138, 5, 206, ..., 31, 43, 244],
[ 29, 134, 131, ..., 100, 107, 1],
...,
[109, 166, 14, ..., 64, 95, 102],
[152, 169, 102, ..., 240, 166, 148],
[ 47, 14, 129, ..., 237, 11, 78]], dtype=uint16)
int32
現在、 sがsにキャストされているため、これは実際には意味がありuint16
ません。これは、安全な方法ではありません。
>>> np.can_cast('int32', 'uint16')
False
lut
のdtypeを、、、、uint16
またはuint32
のいずれuint64
かに設定するとコードは機能しますが、、およびの場合は失敗します。int32
int64
uint8
int8
int16
私は何かが足りないのですか、それともこれは単にnumpyで壊れていますか?
回避策も歓迎します...LUTはそれほど大きくないので、2倍のスペースが必要な場合でも、その型を配列の型と一致させることはそれほど悪くないと思いますが、それを行うのは正しくないと感じています。 ..
キャストの安全性を気にしないようにnumpyに指示する方法はありますか?