isinstance(obj, list) 是判断对象是否为 list 的最准确方法,它检查实际类型而非可迭代性或行为;需严格排除子类时用 type(obj) is list。
isinstance(obj, list) 是最直接的方式很多人误以为 hasattr(obj, '__iter__') 或 iter(obj) 不报错就能判断是 list,其实那只是说明它是可迭代对象(比如 str、tuple、set、甚至自定义类),和是不是 list 完全无关。
isinstance(obj, list) 检查的是对象的**实际类型**,不是鸭子类型。它不会把 tuple 或 deque 当成 list,也不会被重载了 __iter__ 的类欺骗。
isinstance([1, 2], list) → True
isinstance((1, 2), list) → False(哪怕它能 for 循环)isinstance(list(), list) → True,但 isinstance(list_subclass(), list) 也返回 True(子类也算)type(obj) is list
如果业务逻辑要求「必须是原生 list 类型,不能是继承自 list 的子类」,就得用身份比较:type(obj) is list。
它比 isinstance 更严格,因为 type() 返回的是对象的**精确类型对象**,而 is 是内存地址级比较。
type([1, 2]) is list → True
type(list_subclass()) is list → False(即使它继承自 list)MyList = list,那么 type(MyList()) is list 仍为 True(因为只是别名,没新建类型)obj.__class__ == list 或 str(type(obj))
这两种写法看似能绕过 isinstance 的继承逻辑,但都不可靠:
obj.__class__ == list:在极少数动态修改 __class__ 的场景下可能失效(虽然罕见,但不符合类型检查本意)str(type(obj)) == "" :依赖字符串解析,受 Python 版本、实现(如 PyPy)、甚至自定义 __repr__ 干扰,完全不推荐from typing import List; isinstance([], List) 返回 False —— List 是类型提示,不是运行时类型collec
tions.abc.Sequence?因为这不是“判断是否是 list”,而是“判断是否像 list 一样支持索引和切片”。str、tuple、range、array.array 都满足 isinstance(obj, Sequence),但它们显然不是 list。
如果你真正关心的是「能否用 obj[0]、obj[1:3]、len(obj)」,那用 Sequence 合理;但标题明确要“真的是 list”,那就不能退让。
类型检查的边界就在这里:运行时判断类型,不是判断行为。想安全地调用 obj.append() 或 obj.sort()?只有 list 和它的子类才保证有这些方法——而 isinstance(obj, list) 就是那个最小、最准、最无歧义的判断依据。