栏目分类:
子分类:
返回
名师互学网用户登录
快速导航关闭
当前搜索
当前分类
子分类
实用工具
热门搜索
名师互学网 > IT > 软件开发 > 后端开发 > Python

Python日志记录装饰器实现

Python 更新时间: 发布时间: IT归档 最新发布 模块sitemap 名妆网 法律咨询 聚返吧 英语巴士网 伯小乐 网商动力

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__)

转载请注明:文章转载自 www.mshxw.com
本文地址:https://www.mshxw.com/it/422990.html
我们一直用心在做
关于我们 文章归档 网站地图 联系我们

版权所有 (c)2021-2022 MSHXW.COM

ICP备案号:晋ICP备2021003244-6号