Python日志记录装饰器实现

由来

需求:为一个加法函数增加记录实参的功能。

def add(x, y):
	print('add called. x={}, y={}'.format(x, y)) # 增加的记录功能
return x + y
add(4, 5)

上面的代码满足了需求,但有缺点:
记录信息的功能,可以是一个单独的功能。显然和add函数耦合太紧密。加法函数属于业务功能,输出信息属于非功能代码,不该放在add函数中

日志记录装饰器实现

import time
import datetime

def logger(wrapped):
	def wrapper(*args, **kwargs):
		start = datetime.datetime.now()
		ret = wrapped(*args, **kwargs)
		delta = (datetime.datetime.now() - start).total_seconds()
		print('{} tooks {}s'.format(wrapped.__name__, delat))
		
		return ret
	return wrapper

@logger  #等价于add = logger(add)
def add(x, y):
	time.sleep(1)
	return x + y

print(add(4, 5)) ##等价于 logger(add)(4, 5)

被装饰后,函数名和文档都不对了。如何解决?

functools模块提供了一个wraps装饰器函数,本质上调用的是,update_wrapper,它就是一个属性复制函数。
wraps(wrapped, assigned=WRAPPER_ASSIGNMENTS, updated=WRAPPER_UPDATES)
wrapped就是被包装函数
wrapper就是包装函数
用被包装函数的属性覆盖包装函数的同名属性
元组WRAPPER_ASSIGNMENTS中是要被覆盖的属性
module , name , qualname , doc , annotations
模块名、名称、限定名、文档、参数注解

import time
from datetime import datetime
from functools import wraps


# def update(wrapped, wrapper):
#     wrapper.__name__ = wrapped.__name__
#     wrapper.__doc__ = wrapped.__doc__


def logger(wrapped):
    @wraps(wrapped)
    def wrapper(*args, **kwargs):
        start = datetime.now()
        ret = wrapped(*args, **kwargs)
        delat = (datetime.now() - start).total_seconds()
        print('{} tooks {}s'.format(wrapped.__name__, delat))
        return ret

    # update_wrapper(wrapper, wrapped)
    return wrapper


@logger # add = logger(add)
def add(x, y:int):
    "add description"
    # time.sleep(2)
    return x + y


print(add(4, 5))
print(add.__name__, add.__doc__, add.__annotations__)

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值