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

hmoban主题是根据ripro二开的主题,极致后台体验,无插件,集成会员系统
自学咖网 » Python之被装饰函数参数的设置与定义