self是实例方法的第一个形参,由Python解释器在通过实例调用方法时自动绑定当前对象;它非关键字,但为约定俗成的命名惯例,绑定发生在调用时刻而非定义时刻,依赖描述符协议实现。

Python实例方法调用时,self 参数不是你手动传入的,而是由解释器在调用时自动绑定到当前实例对象的——这是方法绑定机制的核心。
实例方法本质是“未绑定函数”
定义在类中的普通方法,在类层面看只是一个普通函数对象,它并不属于某个具体实例。只有通过实例调用时,Python 才会将该函数“绑定”为实例方法,并把实例作为第一个参数(即 self)自动传入。
- 例如:
obj.method()等价于Class.method(obj) - 直接用
Class.method(obj)调用时,必须显式传入实例,否则会报TypeError: missing 1 required positional argument: 'self' - 若用
obj.method(不加括号),得到的是一个已绑定的bound method对象,此时 self 已经被“记住”了
self 是约定俗成的参数名,不是关键字
self 本身不是 Python 关键字,只是广泛遵循的命名惯例。你可以写成 this、me 甚至 banana,只要位置正确(第一个形参),解释器就把它当作实例引用。
绑定过程发生在调用时刻,而非定义时刻
绑定不是在类定义时完成的,而是在每次通过实例访问方法属性时动态发生的。这个过程由描述符协议(__get__ 方法)实现:
立即学习“Python免费学习笔记(深入)”;
- 当访问
obj.method,Python 触发函数对象的__get__(func, obj, cls) - 若
obj非空,返回一个绑定了obj的bound method - 若通过类访问(
Class.method),则返回原函数(unbound function,Python 3 中已统一为普通函数)
常见误区与调试提示
很多初学者报错 TypeError: method() takes 1 positional argument but 2 were given,往往是因为:
- 忘记在方法定义中写 self,却用实例调用了它
- 误把实例方法当成静态方法使用,比如在装饰器中未正确处理绑定逻辑
- 用
functools.partial或 lambda 包装实例方法后丢失了自动绑定行为 - 继承中重写了方法但没调用
super().method(),导致父类方法里的 self 指向异常
理解 self 的绑定本质,能帮你更准确地设计 API、调试方法调用链,也能更自然地过渡到理解类方法(@classmethod)、静态方法(@staticmethod)和描述符等高级机制。










