第9章:符合Python风格的对象-可散列的自定义对象 Vector2d

按照定义,目前 Vector2d 实例是不可散列的,因此不能放入集合( set )中:

class Vector2d:
    
    def __init__(self, x, y):
        self.x = float(x)
        self.y = float(y)
    
    # 同前章节内容
    ......


v1 = Vector2d(3, 4)

hash(v1)

# Traceback (most recent call last):
#   File "/Users/...", line 97, in <module>
#     print(hash(v1))
# TypeError: unhashable type: 'Vector2d'

为了把 Vector2d 实例变成可散列的,必须实现 __hash__ 方法( 还需要 __eq__ 方法,前面已经实现了 )。

目前,我们可以为分量赋新值,如 v1.x = 7Vector2d 类的代码并不阻止这么做,为此,我们要先把 x y 属性设为只读特性:

class Vector2d:

    def __init__(self, x, y):
        # 使用两个前导下划线,把方法或属性标记为私有的。
        self.__x = float(x)
        self.__y = float(y)

    # @property 装饰器把读值方法标记为特性,读值方法与公开属性同名,都是 x。
    @property
    def x(self):
        return self.__x

    @property
    def y(self):
        return self.__x
    
    # 省略重复内容
    ...

然后再实现 __hash__() :

# 在Vector2d类中定义
def __hash__(self):
    hash_ = super(Vector2d, self).__hash__()
    return hash_

然后,Vector2d 实例就变成可散列的了:

v1 = Vector2d(3, 4)
v2 = Vector2d(5, 6)

print(hash(v1))  # -9223372036572724136
print(hash(v2))  # 282051676

⚠️ 总结:要想创建可散列的类型,不一定要实现特性,也不一定要保护实例属性。只需正确地实现 __hash____eq__ 方法即可。但 是,实例的散列值绝不应该变化,因此我们借机提到了只读特性。

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值