Python类中特殊方法__eq__和__hash__的关系

class Point(object):

    def __init__(self, x, y):
        self.x = x
        self.y = y

    def __repr__(self):
        return repr((self.id, self.x, self.y))

    def __eq__(self, other):
        return self.x == other.y and self.y == self.y

    def __hash__(self):
        return hash((self.x, self.y))
  • 上面定义了一个二维点的类其中__repr__主要用来以一个字符串表示该类的实例,例如Point(1,2),在调试时打印该点会获得字符串(1,2)
  • 当对两个点的实例进行值的比较时,比如p1=Point(1,1) p2=Point(1,2),判断p1==p2__eq__()会被调用,用以判断两个实例是否相等。在上述代码中定义了只要x和y的坐标相同,两个点相等。需要注意,__eq__()is不生效,==是比较的值,而is比较的是引用,也就是内存地址。举个例子,p1=Point(1,1) p2=Point(1,1),p1==p2True,p1 is p2False,只有p1 is p1True
  • 在Python中对象分为可哈希对象和不可哈希对象,可哈希对象如字符串、数字、自定义的类、frozenset、元组,被称作不可变对象,不可哈希对象如字典、列表、集合,被称作可变对象。这里的不可变不是对象的值不可变,而是指对象创建后其hash值在其生命周期内不会改变。用函数hash()取可哈希对象的hash值,只要是同一对象其hash值不会改变;而对不可哈希对象取hash值,例如对列表取hash值,会报错,返回TypeError: unhashable type: 'list'。可哈希对象因其hash值不变可以用作字典的key,而不可哈希对象则不行。
    在这里插入图片描述
    在这里插入图片描述
    在这里插入图片描述在这里插入图片描述
  • 当需要对类的一个实例取其hash值时,会调用__hash__()。一般来说,会把实例的所有属性打包成元组,返回其hash值,从而实现自定义__hash__()。在用set()去重时就是对比hash值是否一样,如果两个对象hash值一样代表重复。
  • 用户定义的类默认带有__eq__()__hash__()方法;使用它们与任何对象(自己除外)比较必定不相等,并且 x.__hash__()会返回一个恰当的值以确保 x == y 同时意味着 x is yhash(x) == hash(y)
  • 如果一个类没有定义__eq__()方法,那么也不应该定义 __hash__()操作;如果它定义了__eq__()但没有定义 __hash__(),那么__hash__()会被隐式地设为None,这个类就变成了不可哈希对象。如果一个类定义了可变对象并实现了 __eq__()方法,则不应该实现__hash__(),因为可哈希集的实现要求键的哈希集是不可变的。例如,Point类中添加一个属性li是一个列表,由于列表不可哈希所以强行放入包含属性的元组中并返回其哈希值会报错。
    在这里插入图片描述
    在这里插入图片描述
    在这里插入图片描述
  • 如果使用默认的__hash__()则不论如何改变一个实例的值其hash值都不变;反之,使用本文这种自定义的__hash__()方法,实例的值改变后,hash值就会改变。因此,自定义__hash__()方法的类的实例不应该作为字典的key(强行作为key不会报错,但是改变实例的属性值会导致找不到key对应的value),key的哈希值必须唯一不可变,key的hash值改变会导致找不到key对应的value。
    在这里插入图片描述
    在这里插入图片描述
    在这里插入图片描述
    在这里插入图片描述
    在这里插入图片描述
    在这里插入图片描述
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Python中,`__hash__()`是一个特殊方法(也称为魔术方法),用于定义对象的哈希值。哈希值是一个唯一的整数,用于在集合(如字典、集合)中快速查找和比较对象。 当我们在自定义类中没有定义`__hash__()`方法时,默认情况下,对象的哈希值是根据对象的内存地址计算的,即`id(obj)`。这意味着相同的对象可能具有不同的哈希值,因为它们在内存中具有不同的地址。 如果我们想要自定义一个类的哈希值,可以在类中实现`__hash__()`方法。这个方法应该返回一个整数,用于表示对象的哈希值。 下面是一个简单的示例,展示了自定义类中`__hash__()`方法的使用: ```python class Student: def __init__(self, name, age): self.name = name self.age = age def __hash__(self): # 自定义哈希值为年龄的平方 return self.age ** 2 student1 = Student("Alice", 20) student2 = Student("Bob", 25) print(hash(student1)) # 输出:400 print(hash(student2)) # 输出:625 ``` 在上述示例中,我们定义了一个`Student`类,并在其中实现了`__hash__()`方法,计算哈希值为年龄的平方。通过调用`hash()`函数,我们可以获取对象的哈希值并进行打印。 需要注意的是,如果一个类定义了`__hash__()`方法,那么它也必须定义`__eq__()`方法,用于对象的相等性比较。这是因为哈希值只能用于快速查找,而对象的相等性比较需要通过`__eq__()`方法来确定。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值