前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >Python-自定义装饰器,使用装饰器记录函数执行次数,一种埋点的实现形式

Python-自定义装饰器,使用装饰器记录函数执行次数,一种埋点的实现形式

作者头像
学到老
发布2019-01-25 10:39:32
1.3K0
发布2019-01-25 10:39:32
举报

什么是装饰器?

装饰器本质是一个函数,它可以在不改变原来的函数的基础上额外的增加一些功能。如常见的@classmethod,@staticmethod等都是装饰器,接下来记录下如何自定义个装饰器: 刚刚说过了,装饰器的本质就是一个函数,所有想要自定义一个装饰器,首先自定义一个函数

代码语言:javascript
复制
def decorate(func):
    def wrapper(*args,**kwargs):
        print("定义一个装饰器")
        func(*args,**kwargs)
    return wrapper

此时就已经定义好了一个基本的装饰器,那该如何调用呢?

代码语言:javascript
复制
@decorate
def text1():
    print("text1")

text1()

输出的结果为:

代码语言:javascript
复制
定义一个装饰器
text1

分析:此时的@decorate相当于将text1函数的内存地址传入decorate函数,并返回wrapper函数的内存地址。因此在代码结尾中调用text1()本质上是执行wrapper函数。因为执行的是wrapper函数,所以会打印”定义一个装饰器”,又因为func函数是text1的内存地址,所以调用func,会打印”text1”。

带参数的装饰器 上面介绍了一个简单的装饰器如何定义,可是我们常常看到一个装饰器@xxxxxxx(abc=”python”),这种装饰器是如何封装的,原理又是怎么样的呢

代码语言:javascript
复制
def decorate(name):
    def wrapper(func):
        def sub_wrapper(*args,**kwargs):
            print("定义一个带参数的装饰器",name)
            func(*args,**kwargs)
        return sub_wrapper    
    return wrapper

@decorate(name="python")
def text1():
    print("text1")

text1()

输出结果:

代码语言:javascript
复制
定义一个带参数的装饰器 python
text1

分析:带参数的装饰器与普通的装饰器多加了一层,其实就是讲“python”参数传入decorate函数,并返回wrapper函数的内存地址,再将text1函数内存地址传入wrapper函数,并返回了sub_wrapper函数的内存地址。而在代码末尾调用text1,其实本质是调用了sub_wrapper函数。

使用装饰器记录函数执行次数

代码语言:javascript
复制
def set_func(func):
    num = [0]   # 闭包中外函数中的变量指向的引用不可变
    def call_func():
        func()
        num[0] += 1
        print("执行次数",num[0])
    return call_func

# 待测试方法
@set_func
def test():
    pass

test()
test()
test()

# 执行次数 1
# 执行次数 2
# 执行次数 3

使用nonlocal 访问修改外部函数变量

代码语言:javascript
复制
def set_func(func):
    num = 0   # 闭包中外函数中的变量指向的引用不可变
    def call_func():
        func()
        nonlocal num # 使用nonlocal 访问修改外部函数变量
        num += 1
        print("执行次数",num)
    return call_func

# 待测试方法
@set_func
def test():
    pass

test()
test()
test()

# 执行次数 1
# 执行次数 2
# 执行次数 3
本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
原始发表:2018年10月31日,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 什么是装饰器?
  • 使用装饰器记录函数执行次数
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档