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的标准规范。
网友评论