今天在廖雪峰网站的python教程中看到了这样一个题目:
请编写一个decorator,能在函数调用的前后打印出'begin call'和'end call'的日志。
然后看到文章下面有人给出了这样一个答案:
def log(func):
def dec():
print('begin call %s():' % func.__name__)
result = func()
print('end call %s():' % func.__name__)
return result
return dec
@log
def f1():
print('2016-11-8')
f1()
输出结果是这样的:
begin call f1():
2016-11-8
end call f1():
本人作为编程的初学者,感觉输出结果应该是这样的才对:
begin call f1():
end call f1():
2016-11-8
因为按照代码的顺序,最后才return result,所以result的值应该是最后才出现的才对。
于是我将代码中 return result 和 print('end call %s()' % func.__name__)顺序调整了一下,变成这个样子:
def log(func):
def dec():
print('begin call %s():' % func.__name__)
result = func()
return result
print('end call %s():' % func.__name__)
return dec
@log
def f1():
print('2016-11-8')
f1()
输出结果则变成了这个样子:
begin call f1():
2016-11-8
没有后面的 end call f1(): 这一句!!
于是百度之,翻看了好几个文章的解释,最后在知乎上的一位网友的回答让我有点眉目了:
他的答案是这样的:
return 语句就是将结果返回到调用的地方,并把程序控制权一起返回。
比如如果其他地方有一句
num = add(a, b)
return 语句就是把 a + b 返回到 = 右边,并把程序控制权交给这条赋值语句,执行赋值过程。
根据我自己的理解就是,一般情况下 return 语句是函数定义的最后一部分,而 return 语句后的语句是不会执行的,并且退出函数。当然有其他情况,比如这篇文章所说的 return 放在 try 语句块中的情况下,在 if 语句的情况下,等等等等...这里只讨论 return 在这个代码中的作用。
首先, result = func() 这个语句不是简单赋值, 这里还有一个调用的作用,即 result = func() 的功能是 将定义的 func() 函数赋值给 result 并且调用它,所以整个定义语句是这样一个结构:
先执行这句:
print('begin call %s():' % func.__name__)
然后执行这句:
def f1():
print('2016-11-8')
最后执行这句:
print('end call %s():' % func.__name__)
而 return result 的意思是 返回 result 的值,并且返回的值出现在 result 调用的地方。