Python每天3分钟 – #009深入理解is和==的区别

欢迎访问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的标准规范。

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


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

原文链接:http://www.zsiss.com/4920.html,转载请注明出处。

0

评论0

请先

爱分享推出ChatGPT国内镜像,无需魔法直接用!写文章,写代码,做PPT,做网站原创软文效果好到爆炸 https://chat.gcrup.com

社交账号快速登录