在Numpy数组子类中更改__getitem__和__setitem__的行为

时间:2019-01-21 18:09:50

标签: python arrays numpy getter-setter

Numpy arrays can be efficiently subclassed,但我想修改__getitem____setitem__的行为,以便它们可以采用日期时间范围,同时保留诸如操作之类的内置机械的最大数量,等等。可以用__array_ufunc__完成吗?

似乎在他们的example中,numpy.ufunc.at方法被覆盖了。

这可以用来修改numpy数组的获取/设置行为吗?

1 个答案:

答案 0 :(得分:1)

您可以实现__getitem____setitem__来处理您的特定情况(带有日期时间对象),并在其他情况下分派到super().__{get|set}item__。这样,ndarray的其余功能得以保留。例如:

from datetime import date
import numpy as np

class A(np.ndarray):
    def __array_finalize__(self, obj):
        if obj is not None:
            obj.start_date = date.today()

    def __getitem__(self, item):
        if isinstance(item, slice) and isinstance(item.start, date) and isinstance(item.stop, date):
            return super().__getitem__(slice((item.start - self.start_date).days,
                                             (item.stop - self.start_date).days,
                                             item.step))
        return super().__getitem__(item)

a = A((10,), buffer=np.arange(10), dtype=int)
print(a[1:8])
print(a[date(2019, 1, 22):date(2019, 1, 29):2])
print(np.cumsum(a))
print(np.add.outer(a, a))

哪个输出:

[1 2 3 4 5 6 7]
[1 3 5 7]
[ 0  1  3  6 10 15 21 28 36 45]
[[ 0  1  2  3  4  5  6  7  8  9]
 [ 1  2  3  4  5  6  7  8  9 10]
 [ 2  3  4  5  6  7  8  9 10 11]
 [ 3  4  5  6  7  8  9 10 11 12]
 [ 4  5  6  7  8  9 10 11 12 13]
 [ 5  6  7  8  9 10 11 12 13 14]
 [ 6  7  8  9 10 11 12 13 14 15]
 [ 7  8  9 10 11 12 13 14 15 16]
 [ 8  9 10 11 12 13 14 15 16 17]
 [ 9 10 11 12 13 14 15 16 17 18]]