python装饰器详解
本文链接:https://www.cnblogs.com/tujia/p/13144814.html
一句话说明:print( get_something(params) ) 实例上调用的是 print( auth(get_something)(params) )
解释:
1)装饰器本身也是一个函数,并且它的返回值也必须是一个函数;
2)处理过程是把原函数当作参数传入装饰器函数,然后向返回的函数传入原函数的参数并执行,最后得到执行结果
Demo1 装饰器啥也不做,直接返回函数本身:
def auth(func): return func @auth def get_something(): return '大宝贝' print(get_something())
# 大宝贝
注:装饰器返回的函数会被自动执行。
Demo2 装饰器返回了函数,但没处理原函数:
def auth(func): def anonymity(): return '无关的东东' return anonymity @auth def get_something(): return '大宝贝' print(get_something())
# 无关的东东
注:返回anonymity函数会自动执行了,返回 "无关的东东";但 func(即 get_something)没有被处理到。
Demo3 装饰器内部函数返回被装饰函数:
def auth(func): def anonymity(): return func return anonymity @auth def get_something(): return '大宝贝' print(get_something()) # <function get_something at 0x057DB468>
注:这个demo乍看和Demo1是一样的,但其实是不一样的。auth装饰器返回的是 anonymity 函数会自动被执行,但返回的是 func 函数对象,func并没有被执行 。
Demo4 在auth装饰器里的anonmity函数里直接执行 get_something 但没返回结果:
def auth(func): def anonymity(): func() return None return anonymity @auth def get_something(): print('你调用了 get_something 函数') return '大宝贝' print(get_something()) # 你调用了 get_something 函数 # None
注:auth返回anonmity并被自动执行的时候,会调用func函数(也就是get_something函数)输出 ”你调用了 get_something 函数“,但没有接收和处理到 func() 的返回值,所以不会输出“大宝贝”。
Demo5 同上,但返回的不是None,是返回 func 的函数结果:
def auth(func): def anonymity(): return func() return anonymity @auth def get_something(): print('你调用了 get_something 函数') return '大宝贝' print(get_something()) # 你调用了 get_something 函数 # 大宝贝
注: anonymity被执行的时候,执行了 func(),并返回了 func() 的返回值。
Demo6 接收被装饰函数的参数:
def auth(func): def anonymity(token): if token == '123456': return func(token) else: return None return anonymity @auth def get_something(token): return '大宝贝' print(get_something('123456')) print(get_something('456789')) # 大宝贝 # None
注:anonymity接收了token参数但做了判断,判断是否执行 func 函数(原get_something函数)
Demo 7 接收原函数所有参数:
def auth(func): def anonymity(*args, **kwargs): if 'token' in kwargs: token = kwargs['token'] else: token = args[0] if token == '123456': return func(*args, **kwargs) else: return None return anonymity @auth def get_something(token): return '大宝贝' print(get_something('123456')) print(get_something(token='456789')) # 大宝贝 # None
注:关键是 *args 和 **kwargs
Demo8 带参数的装饰器:
def auth(source=''): def wrap(func): def anonymity(*args, **kwargs): token = kwargs['token'] if 'token' in kwargs else args[0] if source == 'app' and token == '123456': return func(*args, **kwargs) elif source == 'm' and token == '456789': return func(*args, **kwargs) else: return None return anonymity return wrap @auth(source='app') def app_get_something(token): return '大宝贝' @auth(source='m') def m_get_something(token): return '大宝贝' print(app_get_something(token='123456')) print(m_get_something(token='456789')) # 大宝贝 # 大宝贝
注:当装饰器增加传入参数时,装饰器要多嵌套一层函数,原函数被当作参数传给装饰器的返回函数,原函数的参数被传入装饰器的返回函数的返回函数
即:auth(source='app')(app_get_something)(token='123456') 、auth(source='m')(m_get_something)(token='456789')
完。