欢迎访问Python每天3分钟系列。

每天早上8点半,花3分钟时间,学习或温习一个Python知识点。

图片

今天是第009篇:深入理解is和==的区别

is和==的区别

  • is是比较两个对象在内存中地址是否相同
  • ==是比较两个对象的值是否相同,它调用的是对象的__eq__方法

用一个自定义类深入理解它们的区别

定义一个学生类,它包含名字(name)和学号(no)两个属性。

class Student:
  def __init__(self, name, no):
    self.name = name
    self.no = no # 学号

来测试一下is和==:

s1 = Student('张三', '9527')
s2 = Student('张三', '9527')
print(id(s1))  #打印s1的内存地址
print(id(s2))  #打印s2的内存地址
print(s1 is s2)
print(s1 == s2)

执行的结果是:

4340284240
4348522368
False
False

函数id是Python内置的函数,用来返回对象的内存地址。

s1和s2虽然name和no都相同,但是它们在内存中是两个不同的对象,所以s1 is s2返回False这很合理。

那为什么s1 == s2也是False呢?==不是比较值吗?

严格说==是调用函数的__eq__()函数。我们的Student类没有定义这个函数,它默认还是比较的内存地址,所以还是False。

我们增强一下这个类:

class Student:
  def __init__(self, name, no):
    self.name = name
    self.no = no # 学号

  # 定义==函数,如果name和no都相同就认为是相等的
  def __eq__(self, other):
   return self.name == other.name and self.no == other.no

s1 = Student('张三', '9527')
s2 = Student('张三', '9527')
print(id(s1))
print(id(s2))
print(s1 == s2)
print(s1 is s2)

现在的打印结果是:

4342365008
4343836736
True
False

因为有了__eq__函数,s1 == s2返回了True。

这个函数的规则可以根据业务需要定义,比如我们规定:只要no相同就认为同一个学生。这种如果学生改名了,也不会影响比较结果。

总之,根据自己的需要去定义这个函数。

结论

  • is是比较两个对象在内存中地址是否相同。
  • ==是比较两个对象的值是否相同,它调用的是对象的__eq__方法。
  • 如果用is是Ture,那么==一定也是True;反过来则不一定。
  • 在大部分情况下,我们是在比较值,所以应该用==。具体用哪个看业务需要,但用错了就会出现很诡异的bug。

再补充一点:

整数,字符串对象,因为Python解释器做了一些性能优化,会重复利用原有的对象,所以用is==的结果可能是相同的。这取决于Python解释器的内存实现,不是Python的标准规范。

今天就说到这里,有问题给我留言。


如果你希望我更新某个特定小知识,欢迎给我留言。

声明:本网站资源来源于网络收集,如有侵权,请联系站长进行删除处理。 分享目的仅供大家学习和交流,请不要用于商业用途,否则后果自负。本站提供的源码、模板、插件等等其他资源,都不包含技术服务请大家谅解。本站资源售价只是赞助,收取费用仅维持本站的日常运营所需。反馈邮箱:1159995880@qq.com