补充 : 语法糖案例调用关系详解
装饰器案例
- 代码
def dec_a(func):
print('111')
def wrapper(*args, **kw):
print('222')
func(*args, **kw)
print('333')
return wrapper
def dec_b(func):
print('aaa')
def wrapper(*args, **kw):
print('bbb')
func(*args, **kw)
print('ccc')
return wrapper
@dec_a
@dec_b
def test():
print('test')
test()
- 返回值结果
aaa
111
222
bbb
test
ccc
333
分析一:单层语法糖引入
- 首先我们要看最后的调用方法,要明白他的调用关系
@dec_a
@dec_b
def test():
print('test')
-
我们看这行代码
-
我们都知道语法糖的本质就是替代了我们原本的返回内部函数调用内存地址的这一步
-
即语法糖语法
@dec_b def test(): print('test') test()
-
正常语法
test = dec_b(test) def test(): print('test') test()
-
-
我们明白单层语法糖,其实多层语法糖也就变得简单了
-
我们只需要明白,每一层返回的语法糖,返回的是什么即可明白调用关系
单层语法糖的调用关系分析
- 代码
def dec_b(func):
print('aaa')
def wrapper(*args, **kw):
print('bbb')
func(*args, **kw)
print('ccc')
return wrapper
test = dec_b(test)
def test():
print('test')
test()
- 返回值结果
aaa
bbb
test
ccc
- 分析
- 调用
dec_b(test)
传进去外部 test 的内存地址
拿到返回的内存空间地址
赋给变量test(返回的内存空间地址)
- 调用
- 调用
- 下一步的前提是已经过了一遍 装饰器 了 ,那也就是走完一遍装饰器内的函数了。也就是拿到了一个打印的
aaa
- 调用
test(返回的内存空间地址)
得到bbb
- 遇到
func
执行外部的test函数
得到test
- 执行剩下的函数 得到
ccc
- 下一步的前提是已经过了一遍 装饰器 了 ,那也就是走完一遍装饰器内的函数了。也就是拿到了一个打印的
分析二:多层语法糖调用关系
- 我们分析我们下面的调用关系的这行代码
@dec_a
@dec_b
def test():
print('test')
-
@dec_b
-
我们先按正常语法分析
dec_b返回的内存空间地址 = dec_b(test)
-
本质上就是调用
dec_b
这个函数内部的调用函数的内存空间地址
-
-
@dec_a
-
我们也按正常语法分析
dec_a返回的内存空间地址 = dec_a(参数?)
-
到这里我们发现我们按照正常语法糖来说,是需要传入一个调用函数名进去,返回给我们一个内存空间地址的
-
在这里如果没有第一层的
@dec_b
那一定是会报错的 -
但是正因为有了第一层语法糖的调用,我们返回了一个待定的内存空间地址(在这步双层语法糖叠加我们没办法给我们返回的内存空间地址赋变量名,默认使用返回的变量名即
@dec_b
的 wrapper) -
那我们就将这个内存地址传回去
-
最后我们需要一个变量名来接受最后的内存空间地址,没有则默认用调用函数的函数名即
test
-
分析三:多层语法糖的调用顺序
-
经过上面的分析我们已经知道了我们的多层语法糖的调用关系
-
即
@dec_a # test(dec_a返回的wrapper)= dec_a(wrapper) @dec_b # dec_b返回的wrapper = dec_b(text) def test(): print('test')
-
-
那执行顺序,我们最后返回的是
test(dec_a的wrapper)
-
函数调用要从最后返回的地址往前返回(前提是我们要拿到内存空间地址)
-
首先我们先执行第一层语法糖
@dec_b
-
我们最先调用
dec_b(text)
拿到dec_b返回的wrapper
- 进入到
dec_b(text)
时,我们就打印了内部的aaa
- 进入到
-
接着我们调用第二层语法糖
@dec_a
-
我们将
dec_b的wrapper
作为参数传进去,就有了dec_a(dec_b返回的wrapper)
- 进入到
dec_a(dec_b返回的wrapper)
就执行了内部的111
- 同时接收到返回的
dec_a的wrapper
- 进入到
-
最后我们用我们默认的函数名来接受这个内存地址,也就是
test = dec_a(dec_b返回的wrapper)
-
分析四:多层语法糖的执行顺序
-
通过上面的分析我们现在的逻辑是
test = dec_a(dec_b返回的的wrapper)
dec_b返回的wrapper = dec_b(text)
-
执行test(其实这里的内存地址并不是原本的test,是已经被语法糖赋予新的内存地址的命名了)
- 去调用
dec_a(wrapper)
得到222
- 遇到
dec_a
内的func()
- 这时的
func
传进来接受到的其实是dec_b的wrapper内存地址
- 这时的
- 遇到
- 于是执行
dec_b
的wrapper
得到bbb
- 此时我们又遇到了
dec_b
内部的func()
- 这时传进来的
func
传进来的接收到的其实是外部的test的内存地址
- 这时传进来的
- 此时我们又遇到了
- 于是执行
test
得到test
- 最后我们要结束我们的调用关系
- 这里其实是最绕的,不明白为什么 又 打印了
ccc
和333
- 这里其实是最绕的,不明白为什么 又 打印了
- 去调用
分析四:最后返回值的分析
-
当我们将我们的执行函数都执行完了以后,我们最后会发现,我们还打印了最后的返回值
ccc
和333
-
我们反过来看,我们的执行顺序
test = dec_a(dec_b返回的的wrapper)
- 执行
test
得到222
- 遇到
func
去执行func
函数- 这里我们注意,我们底下还有一个
print函数
- 当我们调用完我们的
func函数
的时候,我们要走我们剩下的代码,这也就是333
的由来
- 这里我们注意,我们底下还有一个
- 遇到
- 同理
ccc
也是这么来的
-
那为什么是先打印的
ccc
后 打印的333
呢?- 这是因为我们在走到
dec_a
里的func
也就是dec_b 的wrapper
- 转到了
dec_b
里的func
也就是外部的test
- 结束
test
得到 外部的打印结果test
- 结束
dec_b
的func
继续往下走将func
执行完 得到ccc
- 结束
dec_b
回到刚才没结束的dec_a
里的func
- 结束
dec_a
的func
继续往下走 得到了最后的333
- 没有可调用函数关系,结束程序
- 结束
- 这是因为我们在走到
-
返回值结果
aaa # 语法糖第一层 @dec_b的结果
111 # 语法糖第二层 @dec_a的结果
222 # dec_b 返回的 wrapper 函数调用结果(dec_a)内部的 wrapper 函数的执行结果
bbb # dec_a 返回的 wrapper 函数调用结果(dec_b)内部的 wrapper 函数的执行结果
test # 执行到 dec_a 遇到的 func 调用外部 test 函数执行的结果
ccc # 结束dec_b 内的 wrapper 函数调用后执行的结果
333 # 结束dec_a 内的 wrapper 函数调用后执行的结果
本文来自博客园,作者:Chimengmeng,转载请注明原文链接:https://www.cnblogs.com/dream-ze/p/17449507.html