python 带参数装饰器

以下面的廖雪峰老师python教程里装饰器小节留下的练习为例:

写出一个 @log的decorator,使它既支持:

@log
def f():
    pass

 又支持:

@log('execute')
def f():
    pass

参考写法:

import functools

def log(param = None):
    def decorator(fn):
        @functools.wraps(fn)
        def wrap(*args, **kw):
            print('begin call')
            r = fn(*args, **kw)
            print('end call')
            return r
        return wrap
    return decorator if isinstance(param,str) else decorator(param)
    # return decorator

@log
def f():
    print('call f')

@log('execute')
def f2():
    print('call f2')


f()
f2()

运行结果:

begin call
call f
end call
begin call
call f2
end call

 解析:

第一个无参的 @log 的例子,当我们调用 f() 时,调用的其实是 log(f) ,则传给 log 的 param 参数就是函数 f,而 decorator(fn) 函数才是我们真正定义的装饰器,它需要接收一个参数(我们的目标函数),而此时实际执行的确实 log(f),目标函数已经作为参数被 param 接收了,所以我们在 log 函数中要返回 decorator(param),若直接返回 decorator,则没有传递目标函数,报缺少参数的错误。

第二个有参的 @log 的例子,当我们调用 f() 时,调用的其实是 log('execute')(f) ,传给 log 的 param 参数就是 'execute',然后又把目标函数 f 传给返回的 decorator 函数,因此我们在 log 函数最终只要返回 decorator 即可。

也可以这么写:

import functools

def log(param = None):
    def decorator(fn):
        @functools.wraps(fn)
        def wrap(*args, **kw):
            print('begin call')
            r = fn(*args, **kw)
            print('end call')
            return r
        return wrap
    # return decorator if isinstance(param,str) else decorator(param)
    return decorator

@log()
def f():
    print('call f')

@log('execute')
def f2():
    print('call f2')


f()
f2()

 即无论 log 函数传不传参数,在使用 @log的时候都把 () 打上,即若不传参数,则传空值给 param,这样目标函数 fn 就能别装饰器包装到,最后 log 可以统一返回装饰器。

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Python中,装饰器是一种用于修改函数行为的函数。而参数装饰器就是在普通装饰器的基础上,给装饰器传递额外的参数。这样可以在装饰器内部使用这些参数来改变装饰函数的行为。 在给装饰器传递参数时,需要在装饰器函数上面再包裹一层函数来接收参数,并返回装饰器函数。例如,示例代码中的@wrapper_out('QQ')就是一个参数装饰器。 当函数执行到参数装饰器@wrapper_out('QQ')这句时,会先执行wrapper_out('QQ')这个函数,将参数'QQ'传给parameter并得到返回值wrapper函数。然后,再将@符号与wrapper函数结合,得到一个标准的装饰器,按照装饰器的执行流程来装饰函数。 通过参数装饰器,可以在装饰器内部对被装饰函数进行进一步的处理或设置,从而实现更灵活的函数装饰。这对于需要根据不同的参数来选择不同装饰方式的情况非常有用。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *3* [python参数装饰器](https://blog.csdn.net/weixin_44799217/article/details/118695357)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT0_1"}}] [.reference_item style="max-width: 50%"] - *2* [python如何修改装饰器参数](https://download.csdn.net/download/weixin_38570854/12870003)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT0_1"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值