装饰器原理:
装饰器是为了在不改变原函数的功能的同时给原函数增加一些新功能的函数叫做装饰器。
首先说一下简单的装饰器原理
def aa(): print('我就是一个简单的函数')
我们如果再不改变这个函数的基础上添加一个新功能该怎么做?
我们只需要添加一个新函数来调用这个函数,然后在形函数上添加新功能就好了,例如
def decorate(fn): def inner(): print('这个是新加功能') return fn return inner() def aa(): print('我就是一个简单的函数') aa = decorate(aa) aa()
输出结果为:
这个是新加功能 我就是一个简单的函数
下面就看看更详细一点的装饰器
直接上代码,查看结果
def decorate(func): def inner(a,b): print('3 给函数添加一个求和的输出 {} {}'.format(a,b)) print('4 我要打印func函数', func) result = func(a,b) print('6 我要打印add函数', add) print('7 我要打印func函数', func) print('8 我要打印inner函数', inner) return result print('2 我要打印decorate函数',decorate) return inner @decorate def add(a,b): print('5 我要打印add函数', add) return a+b print('1 我应该是首个打印的') print('9 这是函数输出的结果',add(1,2))
输出结果为:
2 我要打印decorate函数 decorate 1 我应该是首个打印的 3 给函数添加一个求和的输出 1 2 4 我要打印func函数 add 5 我要打印add函数 inner 6 我要打印add函数 inner 7 我要打印func函数 add 8 我要打印inner函数 inner 2 我要打印decorate函数 decorate 9 这是函数输出的结果 3
过程分析
由上面的输出,我们可以看得出来,装饰器的执行顺序是,213456789;
那是因为@装饰器函数,然后相当于把函数add传给了装饰器,从而变成了add = decorate(add),这里调用了decorate函数把函数add传到了
装饰器内部,然后就打印了第二步,由于没有后续的调用,程序就在这里在inner处停滞等待被调用,而inner的实际指针已经指向了add函数
所以打我们打印func的时候输出的是add函数,并没有带locals,这里就不属于decorate的内部函数,而我们装饰器下面的add函数却已经成
为了inner的内部函数了,接下来的逻辑就是正常的函数执行逻辑了
#装饰器过程
@decorare+add --》decorate--》inner--》result=func 即 result=add--》
#调用过程
(新add)add(1,2)--》result(1,2)--》(被装饰函数add)add(1,2)--》输出结果
但是经过了装饰器这两个add只是同名而已
免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。