前面讲过,在 Python 的类体中定义的方法默认都是实例方法,也示范了通过对象来调用实例方法。
但要提醒大家的是,Python 的类在很大程度上可看做是一个独立的空间(称为类命名空间),当程序在类体中定义变量、方法时,与前面介绍的定义变量、定义函数其实并没有太大的不同。对比如下代码:
# 定义全局空间的foo函数
def foo ():
print("全局空间的foo方法")
# 全局空间的bar变量
bar = 20
class Bird:
# 定义Bird空间的foo函数
def foo():
print("Bird空间的foo方法")
# 定义Bird空间的bar变量
bar = 200
# 调用全局空间的函数和变量
foo()
print(bar)
# 调用Bird空间的函数和变量
Bird.foo()
print(Bird.bar)
上面代码在全局空间和 Bird 类(Bird 空间)中分别定义了 foo() 函数和 bar 变量,从定义它们的代码来看,几乎没有任何区别,只是在 Bird 类中定义它们时需要缩进。
接下来程序在调用 Bird 空间内的 bar 变量和 foo() 函数(方法)时,只要添加 Bird. 前缀即可,这说明完全可以通过 Bird 类来调用 foo() 函数(方法)。这就是类调用实例方法的证明。
现在问题来了,如果使用类调用实例方法,那么该方法的第一个参数(self)怎么自动绑定呢?例如如下程序:
class User:
def walk (self):
print(self, '正在慢慢地走')
# 通过类调用实例方法
User.walk()
运行上面代码,程序会报出如下错误:
TypeError: walk() missing 1 required positional argument:'self'
请看程序最后一行代码,调用 walk() 方法缺少传入的 self 参数,所以导致程序出错。这说明在使用类调用实例方法时,Python 不会自动为第一个参数绑定调用者。实际上也没法自动绑定,因此实例方法的调用者是类本身,而不是对象。
如果程序依然希望使用类来调用实例方法,则必须手动为方法的第一个参数传入参数值。例如,将上面的最后一行代码改为如下形式:
u = User()
# 显式为方法的第一个参数绑定参数值
User.walk(u)
此代码显式地为 walk() 方法的第一个参数绑定了参数值,这样的调用效果完全等同于执行 u.walk()。
实际上,当通过 User 类调用 walk() 实例方法时,Python 只要求手动为第一个参数绑定参数值,并不要求必须绑定 User 对象,因此也可使用如下代码进行调用:
# 显式为方法的第一个参数绑定fkit字符串参数值
User.walk('fkit')
如果按上面方式进行绑定,那么 'fkit' 字符串就会被传给 walk() 方法的第一个参数 self。因此,运行上面代码,将会看到如下输出结果:
fkit 正在慢慢地走
Python 的类可以调用实例方法,但使用类调用实例方法时,Python 不会自动为方法的第一个参数 self 绑定参数值;程序必须显式地为第一个参数 self 传参,这种方式调用的方法被称为“未绑定方法”。
用类的实例对象访问的类成员方法称为绑定方法;用类名调用的类成员方法称为非绑定方法。