电话
400 9058 355
is 比较对象身份(内存地址),== 调用 eq 方法比较逻辑相等;None 检查必须用 is None;小整数和短字符串缓存属实现细节,不可依赖;自定义类可重载 == 但无法重载 is。
Python 中 is 判断两个变量是否指向**同一块内存地址**(即 id(a) == id(b)),而 == 默认行为是调用左操作数的 __eq__ 方法,由类自己定义“相等”的逻辑。这意味着即使两个对象内容完全一样,is 也可能为 False,而 == 可能为 True。
常见误用场景:用 is 判断字符串或数字是否“值相等”,比如 if x is "hello" 或 if n is 1000——这在某些情况下看似可行,但不可靠。
10 is 10、"a" is "a" 返回 True,但这属于实现细节,不是语言规范保证1000 is 1000 在交互式解释器里可能为 True(因常量折叠),但在函数内或模块中很可能为 False
__eq__,默认继承自 object,此时 == 行为等价于 is
None 是单例对象,全局唯一,所以判断是否为空值时,if x is None 是唯一正确写法。用 == 不仅低效(触发 __eq__ 查找和调用),还可能被恶意重载导致逻辑错误。
例如:
class BadClass:
def __eq__(self, other):
return True # 总是返回 True
x = BadClass()
print(x == None) # True —— 完全违背预期
print(x is None) # False —— 正确
is None,所有主流 linter(如 flake8、pylint)都会对 == None 报 warningNotImplemented、Ellipsis,也应优先用 is
is 是语言级操作,对应 CPython 的

== 底层调用 __eq__,你可以自由实现。这意味着:
__eq__,并通常也要实现 __hash__(如果希望放入 set 或作为 dict 键)__eq__ 时,== 回退到 is,此时二者结果一致__eq__ 时务必注意:返回值必须是布尔值,否则可能引发 TypeError(如返回字符串或 None)示例:
class Point:
def __init__(self, x, y):
self.x, self.y = x, y
def __eq__(self, other):
if not isinstance(other, Point):
return NotImplemented
return self.x == other.x and self.y == other.y
p1 = Point(1, 2)
p2 = Point(1, 2)
print(p1 == p2) # True
print(p1 is p2) # False
单纯比速度:is 确实比 == 快,因为它不涉及方法查找和调用;但这个差距在绝大多数场景下可以忽略(纳秒级)。真正的问题在于语义混淆带来的 bug 难以定位。
is 比较内容是否相同,哪怕它们刚被创建出来且元素一致assert a == b 检查值,而不是 assert a is b,除非你明确想验证对象同一性a == b 为 True 但后续修改一个却没影响另一个——说明它们其实是不同对象,只是值相等,这时用 is 反而能帮你确认是否意外共享了引用最易被忽略的一点:字符串驻留(interning)不是自动对所有字符串生效,只对编译期确定的、符合标识符规则的字符串默认驻留。运行时拼接的字符串,比如 "hello" + "_" + "world",即使结果相同,is 也几乎总是 False。
邮箱:8955556@qq.com
Q Q:8955556
本文详解如何将Go官方present工具(用于生成HTML5...
PySNMP在不同版本中对SNMP错误状态(errorSta...
time.Sleep仅阻塞当前goroutine,其他gor...
PHPfopen()创建含特殊符号的文件名失败主因是操作系统...
WooCommerce中通过代码为分组产品动态聚合子商品的属...
io.ReadFull返回io.ErrUnexpectedE...
本文详解Yii2中控制器向视图传递ActiveRecord数...
本文详解为何通过wp_set_object_terms()为...
Pytest中使用@mock.patch类装饰器会导致补丁泄...
带缓冲的channel是并发安全的FIFO队列;make(c...