python基础之:九步认识装饰器
step1.
先看个代码吧:
def f():print('111111')
f=lambda a:a +100 #覆盖上面的函数f
print(f) #函数名指函数所在内存中的位置,入带后面括号表示执行函数
print(f(10))
out:
<function <lambda> at 0x101b7b6a8>
110
以上code说明:
1.在def 函数时,以顺序执行,如果相同的函数名,会被最后函数覆盖以前的
2.如果直接func名而没有后面的圆括号的话,只指向函数在内存中的位置
3.lambda表达式会自动return结果,而def需要定义return值
step2:
python中可想函数传递参数,并将参数转变为本地变量存在于函数内部。
def f(x):print(locals())
f('a')
out:
{'x': 'a'}
step3:
python函数中可以嵌套函数,这就说明了我们可以在函数里定义函数,而且现有的作用域和函数的变量生存周期依旧适用。
def outer():x='hello world'
def inner():
print(x) #1
inner() #2
outer()
out:
hello world
#1 中发生了什么,函数inner需要一个x的变量,去查找x的变量,但本地变量中没有,查找失败后去上一层的作用域中去寻找,而x变量在上一层函数的作用域中。
对outer来说,x是本地变量,但函数inner可以访问封闭的作用域
#2 中,调用了inner函数,python解释器会优先查找本地的变量x
step4:
def outer():x='hello world'
def inner():
print(x)
return inner #1
foo=outer() #2
print(foo)
foo()
out:
<function outer.<locals>.inner at 0x10137b6a8>
hello world
#1与step3区别,我们将函数inner的内存地址给返回了
#2,将函数outer的返回值赋值于变量foo,也就是说,foo() 执行时,实际上执行了inner()。
但inner函数集成了outer函数中的本地变量,outer外面,没有x这个变量,为什么能执行呢??
这是因为python中有一个叫”函数闭包”的特性,怎么理解呢?嵌套定义在非全局作用域中的函数,能够记住它在被定义时候它所处的封闭命名空间。
函数outer每次在被调用的时候,函数inner都会被重新定义,现在变量x的值不会变化,所以每次执行的逻辑和结果都一样,那么如果x是变化的呢?
step5:
def outer(x):def inner():
print(x) #1
return inner
print1=outer(1)
print2=outer(2)
print1()
print2()
out:
1
2
从这个我们能够看到“闭包”:被函数记住的封闭作用域,能够被用来创建自定义的函数。
事实上,我们并不是传递参数1或2给inner函数,我们实际上是定义了能够打印各种数字的自定义版本。
step6:装饰器
终于说到装饰器上了。。。。。
装饰器其实就是一个闭包而已,把func作为参数,然后返回一个替代版的函数,其实真正执行的是inner内部的流程
def outer(func):def inner():
print('before some_func')
res = func()
print(res+1)
print(('after some_func'))
return res+1
return inner
def foo():
return 1
f1=outer(foo) #1
f1()
out:
before some_func
2
after some_func
在这我们可以认为变量f1是函数foo的的装饰器。但这么写有点太low了是不是,所以python中有个优雅的@符号,太装逼了。。。可以这么写:
def outer(func):def inner():
print('before some_func')
res = func()
print(res+1)
print(('after some_func'))
return res+1
return inner
@outer
def foo():
return 1
foo()
out:
before some_func
2
after some_func
看到没,执行结果一毛一样有木有。
step7:
装饰器用法:
@+函数名
功能:1.自动执行函数(outer函数),并且将下面的函数名f1当做参数
2.将outer函数的返回值,重新赋值给f1
需要注意的两点:
1.有return值得才是完整的装饰器。
2.闭包中返回的inner必须是函数的名称func,并非是func(),如果带了括号,那就是执行函数了, inner中返回为None,因为inner中没有return或者return值为其他了。
step8:
有同志就问了,这特么是一个没有参数的,那如果我自己定义的函数有一个参数咋弄。
那我们就用函数中的万能参数*args **kwargs来造一个通用的装饰器吧:
def outer(func):def inner(*args,**kwargs):
print('before some_func')
res = func(*args,**kwargs)
print(('after some_func'))
return res
return inner
@outer
def foo():
print(111)
foo()
@outer
def foo1(li,d):
print('args:%s'%li)
print('kwargs:%s'%d)
foo1('fuck','U')
===============
out:
before some_func
111
after some_func
before some_func
args:fuck
kwargs:U
after some_func
step9:
多层装饰器
装饰器类似于俄罗斯套娃,可以一层套一层,譬如:
def outer(func):def inner(*args,**kwargs):
print('before some_func')
res = func(*args,**kwargs)
print(('after some_func'))
return res
return inner
@outer1
@outer
def foo1(li,d):
print('args:%s'%li)
print('kwargs:%s'%d)
foo1('fuck','U')
====================
out:
before:this is outer1
before some_func
args:fuck
kwargs:U
after some_func
after:this is outer1 again!
未完待续。。。
以上是 python基础之:九步认识装饰器 的全部内容, 来源链接: utcz.com/z/387505.html