• Python重载运算符实现自定义序列

    Python 可重载的运算符中,有如下几个和序列操作相关的特殊方法:

    • __len__(self):该方法的返回值决定序列中元素的个数。
    • __getitem__(self, key):该方法获取指定索引对应的元素。该方法的 key 应该是整数值或 slice 对象,否则该方法会引发 KeyError 异常。
    • __contains__(self, item):该方法判断序列是否包含指定元素。
    • __setitem__(self, key, value):该方法设置指定索引对应的元素。该方法的 key 应该是整数值或 slice 对象,否则该方法会引发 KeyError 异常。
    • __delitem__(self, key):该方法删除指定索引对应的元素。

    在此基础上,如果根据特定的需求对这些方法进行重载,即可实现自定义一个序列类。

    如果程序要实现不可变序列(程序只能获取序列中的元素,不能修改),只要实现上面前 3 个方法就行;如果程序要实现可变序列(程序既能获取序列中的元素,也可修改),则需要实现上面 5 个方法。

    下面程序将会实现一个字符串序列,在该字符串序列中默认每个字符串的长度都是 3,该序列的元素按 AAA、 AAB、 AAC…… 这种格式排列:

    def check_key (key):
        '''
        该函数将会负责检查序列的索引,该索引必须是整数值,否则引发TypeError
        且程序要求索引必须为非负整数,否则引发IndexError
        '''
        if not isinstance(key, int): raise TypeError('索引值必须是整数')
        if key < 0: raise IndexError('索引值必须是非负整数')
        if key >= 26 ** 3: raise IndexError('索引值不能超过%d' % 26 ** 3)  
    class StringSeq:
        def __init__(self):
            # 用于存储被修改的数据
            self.__changed = {}
            # 用于存储已删除元素的索引
            self.__deleted = []
        def __len__(self):
            return 26 ** 3
        def __getitem__(self, key):
            '''
            根据索引获取序列中元素
            '''
            check_key(key)
            # 如果在self.__changed中找到已经修改后的数据
            if key in self.__changed :
                return self.__changed[key]
            # 如果key在self.__deleted中,说明该元素已被删除
            if key in self.__deleted :
                return None
            # 否则根据计算规则返回序列元素
            three = key // (26 * 26)
            two = ( key - three * 26 * 26) // 26
            one = key % 26
            return chr(65 + three) + chr(65 + two) + chr(65 + one)
        def __setitem__(self, key, value):
            '''
            根据索引修改序列中元素
            '''
            check_key(key)
            # 将修改的元素以key-value对的形式保存在__changed中
            self.__changed[key] = value
        def __delitem__(self, key):
            '''
            根据索引删除序列中元素
            '''
            check_key(key)
            # 如果__deleted列表中没有包含被删除key,添加被删除的key
            if key not in self.__deleted : self.__deleted.append(key)
            # 如果__changed中包含被删除key,删除它
            if key in self.__changed : del self.__changed[key]
    # 创建序列
    sq = StringSeq()
    # 获取序列的长度,实际上就是返回__len__()方法的返回值
    print(len(sq))
    print(sq[26*26])
    # 打印没修改之后的sq[1]
    print(sq[1]) # 'AAB'
    # 修改sq[1]元素
    sq[1] = 'fkit'
    # 打印修改之后的sq[1]
    print(sq[1]) # 'fkit'
    # 删除sq[1]
    del sq[1]
    print(sq[1]) # None
    # 再次对sq[1]赋值
    sq[1] = 'crazyit'
    print(sq[1]) # crazyit

    上面程序实现了一个 StringSeq 类,并为该类实现了一个 __len__()、 __getitem__()、 __setitem__() 和 __delitem__() 方法,其中 __len__() 方法返回该序列包含的元素个数,__getitem__() 方法根据索引返回元素,__setitem__() 方法根据索引修改元素的值,而 __delitem__() 方法则用于根据索引删除元素。

    需要注意的是,该序列本身并不保存序列元素,它会根据索引动态计算序列元素,因此在使用时需要保存使用 __changed 和 __deleted 分别保存被修改和被删除的元素。
     
    在定义了字符串序列之后, 接下来程序创建了序列对象,并调用序列方法测试该工具类。运行该程序,可以看到如下输出结果:

    17576
    BAA
    AAB
    fkit
    None
    crazyit

    从上面的输出结果来看,程序中序列的第二个元素 sq[1] 恰好为 'AAB',程序既可对序列元素赋值,也可删除、修改序列元素,这完全是一个功能完备的序列。

更多...

加载中...