类中定义函数分为了两大类,绑定方法与非绑定方法,它们有一些特殊之处:
1、绑定方法特殊之处:绑定给谁就应该由谁来调用,谁来调用就会将谁当做第一个参数自动传入
绑定给对象的方法:这个在面向对象第一篇第六节就讲到过(传送门),在类中定义函数没有被任何装饰器修饰的情况下,默认就是绑定对象的
绑定给类的方法:为类中定义函数添加一个装饰器 classmethod,就是绑定类的
2、非绑定方法特殊之处:非绑定方法就是一个普通函数,既不与类绑定又不与对象绑定,意味着类与对象都可以调用,但是无论谁来调用都是一个普通函数,没有自动传值效果,非绑定方法就是为类中定义函数添加一个装饰器 staticmethod
类的绑定方法
类中的方法,默认都是绑定给对象使用,所以需要采取一点措施,将类中的绑定方法解除对象绑定关系,进而绑定到类上,在 Python 中,引入了 classmethod 装饰器,将类中的方法绑定到类身上
class People():
@classmethod
def talk(cls):
pass
p = People()
print(People.talk)
# 运行
<bound method People.talk of <class '__main__.People'>>
从上述结果可以看出,加上了一个装饰器,将类中绑定给对象的方法,绑定到类身上了。之前分析过,如果一个方法绑定到谁身上,那么在调用该函数的时候,将自动将该调用者当作第一个参数传递到函数中。但是,绑定到类的方法与绑定到对象方法有一点点不同:
class People:
def __init__(self, name):
self.name = name
@classmethod
def talk(cls):
pass
p = People('qiu')
print(People.talk)
print(p.talk)
# 运行结果
<bound method People.talk of <class '__main__.People'>>
<bound method People.talk of <class '__main__.People'>>
也就是说,当对象在调用类的绑定方法时,也会默认把类当作参数传递进去!所以下面执行正常,并不会因为这个方法绑定到类身上,而对象调用没有传递参数,报错!
class People:
@classmethod
def talk(cls):
pass
p = People()
People.talk()
p.talk()
但是,如果 talk() 没有参数,则下面代码均会报错
class People:
@classmethod
def talk():
pass
p = People()
People.talk()
p.talk()
#报错结果
talk() takes 0 positional arguments but 1 was given
两者报错结果一致,这就说明了,当对象来调用类的绑定方法时,也是自动将类传递进去,并不需遵循函数参数传递的规则。
绑定方法总结:类中方法默认都是绑定给对象使用,当对象调用绑定方法时,会自动将对象作为第一个参数传递进去;而类来调用,则必须遵循函数参数一一对应的规则,有几个参数,就必须传递几个参数。如果一个方法是用了 classmethod 装饰器,那么这个方法绑定到类身上,不管是对象来调用还是类调用,都会将类作为第一个参数传递进去。
非绑定方法
上面说了,类中的方法要么是绑定给对象使用,要么是绑定给类使用,那么有没有不绑定给两者使用的函数?
答案是,当然有,Python 给我们提供了 staticmethod 装饰器,可以解除绑定关系,将一个类中的方法,变为一个普通函数
import hashlib
import time
class MySQL:
def __init__(self, host, port):
self.id = self.create_id()
self.host = host
self.port = port
@staticmethod
def create_id(): # 就是一个普通工具
m = hashlib.md5(str(time.clock()).encode('utf-8'))
return m.hexdigest()
print(MySQL.create_id)
conn = MySQL('127.0.0.1', 3306)
print(conn.create_id)
# 运行结果
<function MySQL.create_id at 0x0000021BEAAE9B70>
<function MySQL.create_id at 0x0000021BEAAE9B70>
从上面的输出结果可以看出,使用了 staticmethod 装饰了一个函数,那么这个函数跟普通函数没有什么区别。既然是普通函数,那么就遵从函数参数传递规则,有几个参数就传递几个参数。