Python之被装饰函数参数的设置与定义
被装饰函数参数的设置与定义
先来看一段代码
import time def timmer(func): def inner(): start_time=time.time() func() end_time=time.time() print("run time: %s " %(end_time-start_time)) return inner @timmer def index(): time.sleep(2) print("welcome to index page") @timmer def home(name): time.sleep(3) print("welcome to %s home page" % name)
如上所示,home函数添加了一个参数,而index函数并没有参数。按照正常的函数的定义与调用方式,调用index函数和home函数的方式应该是下面这种形式:
index() home("python")
然后我们运行程序就会发现,程序抛出了异常。
File "E:python_learnpy_code est.py", line 28, in <module> home("python") TypeError: inner() takes 0 positional arguments but 1 was given
说个异常说明inner函数不需要位置参数,但是我们给了一个位置参数。回到timmer装饰器定义的部分,可以看到,timmer装饰器的内部函数确实没有定义参数。这样一来,timmer装饰器只能用于装饰没有参数的函数了,我们可以在timmer装饰器定义的时候为inner函数添加一个参数。
import time def timmer(func): def inner(name): start_time=time.time() func(name) end_time=time.time() print("run time: %s " %(end_time-start_time)) return inner @timmer def index(): time.sleep(2) print("welcome to index page") @timmer def home(name): time.sleep(3) print("welcome to %s home page" % name) index() home("python")
但是这样一来,timmer装饰器装饰index函数的时候又会抛出异常,因为index函数没有参数。
File "E:python_learnpy_code est.py", line 27, in <module> index() TypeError: inner() missing 1 required positional argument: 'name'
相关推荐:《Python相关教程》
在不知道被装饰函数的参数个数的情况下,即被装饰函数的参数可变长,且形式不固定的时候,可以使用*args和**kwargs,把上面的代码修改:
import time def timmer(func): def inner(*args,**kwargs): start_time=time.time() func(*args,**kwargs) end_time=time.time() print("run time: %s " %(end_time-start_time)) return inner @timmer def index(): time.sleep(2) print("welcome to index page") @timmer def home(name): time.sleep(3) print("welcome to %s home page" % name) index() home("python")
再次运行程序,查看运行结果:
welcome to index page run time: 2.0 welcome to python home page run time: 3.0
由上可知,在不知道被装饰函数的参数个数时,可以使用*args和**kwargs来表示任意长度任意形式的参数。
被装饰函数的返回值
修改上面的代码,为home函数定义一个返回值,分别打印index函数和home函数的返回值。
import time def timmer(func): def inner(*args,**kwargs): start_time=time.time() func(*args,**kwargs) end_time=time.time() print("run time: %s " %(end_time-start_time)) return inner @timmer def index(): time.sleep(2) print("welcome to index page") @timmer def home(name): time.sleep(3) print("welcome to %s home page" % name) return("home func") index_res=index() print(index_res) home_res=home("python") print(home_res)
运行程序,可以看到:
welcome to index page run time: 2.0 None welcome to python home page run time: 3.0 None
可以看到,home函数中定义的返回值并没有被打印出来,显示的值为None。因为这里执行的home函数不是原始定义的home函数,而是wrapper函数的执行结果。因为wrapper函数并没有定义返回值,所以执行被装饰后的home函数并没有打印出返回值。
修改代码,在timmer装饰器中定义并返回被装饰函数执行的返回值。
import time def timmer(func): def inner(*args,**kwargs): start_time=time.time() res=func(*args,**kwargs) end_time=time.time() print("run time: %s " %(end_time-start_time)) return res return inner @timmer def index(): time.sleep(2) print("welcome to index page") @timmer def home(name): time.sleep(3) print("welcome to %s home page" % name) return("home func") index_res=index() print(index_res) home_res=home("python") print(home_res)
再次执行函数,查看执行结果:
welcome to index page run time: 2.0 None welcome to python home page run time: 3.0 home func
可以看来,原始home函数中定义的返回值被打印出来了。
结论:
(1)如果被装饰函数没有定义返回值,timmer装饰器装饰后的返回值为None。
(2)如果被装饰函数定义了返回值,则timmer装饰器装饰后则返回被装饰函数的返回值。
来源:PY学习网:原文地址:https://www.py.cn/article.html