30. 派生不可变类型

自定义一种新类型的元组,对于传入的可迭代对象,只保留其中int类型且值大于0的元素,例如:

IntTuple([1, -1, 'abc', 6, ['x', 'y'], 3]) => (1, 6, 3)

要求:继承内置tuple类实现IntTuple类。

解决方案:

继承内置tuple类,并实现__new__()方法,在其中修改实例化行为。


  • 对于类的实例化:

在创建类的实例时,会在调用__init__()方法前调用__new__()方法。所有类的实例都是由基类object.__new__()方法来创建的,由基类object.__init__()方法来初始化的。

>>> list('abc')['a', 'b', 'c']>>> l = list.__new__(list, 'abc')>>> l[]>>> list.__init__(l, 'abc')>>> l['a', 'b', 'c']

>>> tuple('abc')('a', 'b', 'c')>>> t = tuple.__new__(tuple, 'abc')>>> t('a', 'b', 'c')>>> tuple.__init__(t, 'abc')>>> tuple.__init__ is object.__init__True

对于tuple类来说,实例是在tuple.__new__()中创建并初始化的,tuple.__init__()并未起到初始化作用。


  • 方案示例:
class IntTuple(tuple):
    
    def __new__(cls, iterable):
        #过滤iterable
        f_it = (e for e in iterable if isinstance(e, int) and e > 0)
        return super().__new__(cls, f_it)               #调用父类的__new__()方法int_t = IntTuple([1, -1, 'abc', 6, ['x', 'y'], 3])print(int_t)(1, 6, 3)               #结果


上一篇:ECS架构笔记


下一篇:java基础总结(八十七)--Ack机制