python 之内置模块

1.用 functools.wraps 定义函数修饰器

  • 装饰器可以对函数进行封装,但是会改变函数信息

  • 使用 functools 的 warps 可以解决这个问题

装饰器

def trace(func):
    def wrapper(*args, **kwargs):
        result = func(*args, **kwargs)
        print('%s(%r, %r) -> %r' % (func.__name__, args, kwargs, result))
        return result
    return wrapper

@trace  
# 装饰器含义就是 fibonacci = trace(fibonacci)
def fibonacci(n):
    """返回第 n 个斐波那契数字。"""
    if n in (0, 1):
        return n
    return (fibonacci(n-2) + fibonacci(n-1))
fibonacci(3)
>>>
fibonacci((1,), {}) -> 1
fibonacci((0,), {}) -> 0
fibonacci((1,), {}) -> 1
fibonacci((2,), {}) -> 1
fibonacci((3,), {}) -> 2

print(fibonacci)
>>>
<function trace.<locals>.wrapper at 0x0000021B0C068700>

trace 函数所返回的值,是它内部定义的那个 wrapper。而我们又用 trace 来修饰原有的 fibonacci 函数,于是,Python 就会把修饰器内部的那个 wrapper 函数,赋值给当前模块中与原函数同名的 fibonacci 变量。对于调试器对象序列化器等需要使用内省机制的那些工具来说,这样的行为会干扰它们的正常运作。会使help函数失效:help(fibonacci)。

warps 本身也是修饰器,它可以帮助开发者编写其他修饰器。将 warps 修饰器运用到 wrapper 函数之后,它就会把与内部函数相关的元数据全部复制到外围函数。

from functools import *

def trace(func):
    @wraps(func)
    def wrapper(*args, **kwargs):
        result = func(*args, **kwargs)
        print('%s(%r, %r) -> %r' % (func.__name__, args, kwargs, result))
        return result
    return wrapper

@trace  
# 装饰器含义就是 fibonacci = trace(fibonacci)
def fibonacci(n):
    """返回第 n 个斐波那契数字。"""
    if n in (0, 1):
        return n
    return (fibonacci(n-2) + fibonacci(n-1))
print(fibonacci)
>>>
<function fibonacci at 0x0000026E0A9DE3A0>

help(fibonacci)
>>>
Help on function fibonacci in module __main__:

fibonacci(n)
    返回第 n 个斐波那契数字。

2.考虑以 contextlib 和 with 语句来改写可复用的 try/finally 代码

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值