浅析Python装饰器

1、什么是装饰器

在介绍装饰器之前,我们先来思考一个问题:使用Python语言进行程序设计时,如果我们想扩展一个函数的功能,一般会怎么做呢?

比如,有一个名为print_info函数,当前该函数内只做一些简单的打印操作,现在我们想扩展这个函数功能,如在发生错误时,我们将错误行号传入到该函数打印出来。

def print_info():
    print("Hello World")

看到这个问题,我们的第一反应肯定是重新修改这个函数。在print_info函数增加一个参数,用于接收外部传入的错误行号,在函数内部增加打印行号信息语句。

def print_info(err_line):
    print("Hello World")
    print("The Error Occurred Line Number: %d" %(err_line))

似乎,通过修改函数内容,也能实现我们的需求。但有没有这种可能呢,不直接进行修改print_info函数内容,通过其他的方式增加print_info函数的额外功能。答案肯定是有的,能实现这种功能的就是我们要讲的装饰器。

装饰器到底是什么呢?使用比较严谨的语言来描述:

装饰器本质上是一个Python函数,它可以让其他函数在不需要做任何代码变动的前提下增加额外功能,装饰器的返回值也是一个函数对象。它经常用于有切面需求的场景,比如:插入日志、性能测试、事务处理、缓存、权限校验等场景。装饰器是解决这类问题的绝佳设计,有了装饰器,我们就可以抽离出大量与函数功能本身无关的雷同代码并继续重用。

概括的讲,装饰器的作用就是为已经存在的函数或对象添加额外的功能

2、怎么写一个装饰器

我们先来看一段这样的代码

1 import sys
 2
 3 def log(fcn):
 4     def wrapper(*argc, **kw):
 5         #New Operations Are Added Here
 6         if len(argc) != 1:
 7             print('Illegal parameter')
 8             return -1
 9         print("The Error Occurred Line Number: %d" %(argc[0]))
10         return fcn()
11     return wrapper
12
13 def print_info():
14     print("Hello World")
15
16 print_info = log(print_info) # 添加功能并保持原函数名不变
17
18 def main():
19     print_info(sys._getframe().f_lineno) # sys._getframe().f_lineno代表当前的行号
20
21 if __name__ == "__main__":
22     main()

log函数是一个“闭包”(关于什么是“闭包”可参考“浅析Python闭包”),该函数有一个参数fcn,返回值是内部实现的wrapper函数。

比较重要的一条语句print_info = log(print_info),这里通过对print_info变量赋值改变它原来指向的类型,起到了对print_info函数赋予了新功能。执行这句语句之后print_info变量不再表示前面定义的print_info函数,而是表示log函数内实现的wrapper函数,但原来定义的print_info函数仍然存在。

最终调用的print_info函数,其实调用的是log中的wrapper函数。由于“闭包”特性,虽然wrapper函数已经离开了创造它的环境log函数,但它仍然可以使用log函数中的自由变量。wrapper函数中,打印了传入的错误行号,并调用了传入的原定义的print_info函数,这样便做到了对定义的print_info函数内容做修改,调用print_info增加了额外功能。

运行结果

The Error Occurred Line Number: 19 # 打印出新增的对应行号信息,在19行被调用,所以传入行号是19
Hello World #原print_info函数执行内容

上面的log函数其实已经是一个装饰器了,它对原函数做了包装并返回了另外一个函数,额外添加了一些功能。因为这样写实在不太优雅,新版本Python中支持了@语法糖,我们可以把它使用起来。

下面代码等同于上面的写法。

import sys

def log(fcn):
    def wrapper(*argc, **kw):
        #New Operations Are Added Here
        if len(argc) != 1:
            print('Illegal parameter')
            return -1
        print("The Error Occurred Line Number: %d" %(argc[0]))
        return fcn()
    return wrapper    

@log
def print_info():
    print("Hello World")

#print_info = log(print_info)

def main():
    print_info(sys._getframe().f_lineno)

if __name__ == "__main__":
    main()

把@log放到print_info函数定义处,相当于执行了print_info = log(print_info)语句

3、完整装饰器的写法

前面写的装饰器都没有问题,但是还差最后一步,虽然装饰器装饰过的函数看上去名字没变,其实已经变了。

def log(fcn):
    def wrapper(*args, **kw):
        print(fcn.__name__)
        return fcn()
    return wrapper  

@log
def print_info():
    print('Hello World')

def main():
    print_info()
    print(print_info.__name__)

if __name__ == "__main__":
    main()

运行结果

print_infoHello Worldwrapper>>>

上面print_info函数经log装饰器装饰后,它的__name__属性已经从原来的'print_info’变成了'wrapper’

因为返回的那个wrapper()函数名字就是'wrapper',所以,需要把原始函数的__name__等属性复制到wrapper()函数中,否则,有些依赖函数签名的代码执行就会出错。

不需要在内部添加wrapper.__name__=fcn.__name__这样的代码,Python内置的functools.wraps就是干这个事的,所以,一个完整的decorator的写法如下:

import functools

def log(fcn):
    @functools.wraps(fcn)
    def wrapper(*args, **kw):
        print(fcn.__name__)
        return fcn()
    return wrapper

4、高阶一点装饰器

4.1 带参数装饰器

如果装饰器函数本身要传入参数,那么装饰器就会是这样的

import sys

def log(text):
    def wrapper(fcn)
        def inner_wrapper(*argc, **kw):
            #New Operations Are Added Here
            print(text)
            if len(argc) != 1:
                print('Illegal parameter')
                return -1
            print("The Error Occurred Line Number: %d" %(argc[0]))
            return fcn()
        return inner_wrapper
    return wrapper    

@log('With Parameter Decorator')
def print_info():
    print("Hello World")

#print_info = log(print_info)

def main():
    print_info(sys._getframe().f_lineno)

if __name__ == "__main__":
    main()

对于带参数装饰器,可以这么理解,当带参数的装饰器被装饰在某个函数上时,比如上述代码@log('With Parameter Decorator')log('With Parameter Decorator')其实是一个函数,会马上被执行,它返回的结果任是一个装饰器wrapper。

也就是下面代码其实等价于print_info = wrapper(print_info)

@log('With Parameter Decorator')
def print_info():
    print("Hello World")

4.2 装饰器装饰类方法

前面我们介绍的都是用装饰器装饰函数,装饰器同时也可以装饰类方法,下面来看如何用装饰器装饰类的方法

我们定义了名为simple_test的类,这个类有一个data属性和get_data方法。类里的实现非常简单,get_data方法只返回data属性值,并不进行其他操作。同时,我们还需定义了一个装饰器装饰类的get_data方法,装饰器中根据数据获取源来编写其中功能,比如我们这里选择从串口终端获取数据,直接修改装饰器的内容让其获取终端输入数据。

这样在设计一个类时,做到了将类中稳定的部分和经常要修改的部分独立开。类设计完成后,我们不需要更改类的源码,在使用时只需要根据需求修改类方法的装饰器就能实现我们的需求。

def serial_data(fcn):
    def wrapper(self, *args, **kw):
        str = input('Please enter an integer: ')
        try:
            tmp = int(str)
        except ValueError:
            print('Invalid Value')
            return None
        self.data = tmp
        return fcn(self)
    return wrapper

class simple_test(object):
    def __init__(self, _val = 0):
        self.data = _val

    @serial_data
    def get_data(self):
        return self.data

def main():
    obj = simple_test()
    val = obj.get_data()
    if val is not None:
        print('get Data: %d' %(val))

if __name__ == "__main__":
    main()

运行结果:

Please enter an integer: 100
get Data: 100
>>>
(0)

相关推荐

  • python笔记36-装饰器之wraps

    前言 前面一篇对python装饰器有了初步的了解了,但是还不够完美,领导看了后又提出了新的需求,希望运行的日志能显示出具体运行的哪个函数. name和doc __name__用于获取函数的名称,__d ...

  • python笔记35-装饰器

    前言 python装饰器本质上就是一个函数,它可以让其他函数在不需要做任何代码变动的前提下增加额外的功能,装饰器的返回值也是一个函数对象. 很多python初学者学到面向对象类和方法是一道大坎,那么p ...

  • 一文看懂Python系列之装饰器(decorator)(工作面试必读)

    Python的装饰器(decorator)可以说是Python的一个神器,它可以在不改变一个函数代码和调用方式的情况下给函数添加新的功能.Python的装饰器同时也是Python学习从入门到精通过程中 ...

  • 初识装饰器函数

    我之前看装饰器文章介绍,很少有用 装饰器函数这种称谓的.但是今天大邓简单的学了下装饰器,觉得应该先让大家知道装饰器是一种函数,让大家从熟悉的函数去学习装饰器.大邓姑且草率的将其称呼为 装饰器函数 装饰 ...

  • Python学习—装饰器

    学习Python已经有一段时间了,陆续学了一些基础部分,但是理解的不是很深刻,每过一段时间就会忘记,所以不得不写一些博客进行记录,加深自己的理解.这两个星期一直在研究装饰器,开始觉得很简单,但是只知其 ...

  • 【进阶】一文读懂Python装饰器,搞清来龙去脉!

    (给机器学习算法与Python学习加星标,提升AI技能) 选自pouannes.github.io 作者:Pierre Ouannes 本文由机器之心(nearhuman2014)翻译 原文:http ...

  • 推荐8个炫酷的 Python 装饰器!

    Python 编程语言的一大优点是它把所有功能都打包到一个小包中,这些功能非常有用.许多特性可以完全改变 Python 代码的功能,这使得该语言更加灵活.如果使用得当,其中一些功能可以有效缩短编写程序 ...

  • 说说对于Python装饰器的理解?

    公众号新增加了一个栏目,就是每天给大家解答一道Python常见的面试题,反正每天不贪多,一天一题,正好合适,只希望这个面试栏目,给那些正在准备面试的同学,提供一点点帮助! 小猿会从最基础的面试题开始, ...

  • 人人都能看懂的 Python 装饰器入门教程!

    很多人认为理解了装饰器的概念和用法后,会觉得自己的 Python 水平有一个明显的提高. 但很多教程在一上来就会给出装饰器的定义以及基本用法,例如你一定会在很多文章中看到例如代码运行时间计时器等相关常 ...

  • Python|装饰器

    一对象的概念python的所有内容都可以作为对象,这意味着这些内容可以作为参数作用于其他的"对象",这不难理解,就像在函数中,可以把另一个函数作为参数,甚至是类作为参数,,因此经常 ...

  • 一文搞懂Python装饰器

    一.前言 本不打算专门写文来讲装饰器的,但有不少粉丝问到了,自己查阅了一些网上的装饰器教程,发现讲的通俗易懂的不多,也有不少照搬的文章.所以我这里专门来讲一讲它. 个人在用的人工智能学习网站推荐给大家 ...

  • Python 中的函数装饰器和闭包

    函数装饰器可以被用于增强方法的某些行为,如果想自己实现装饰器,则必须了解闭包的概念. 装饰器的基本概念 装饰器是一个可调用对象,它的参数是另一个函数,称为被装饰函数.装饰器可以修改这个函数再将其返回, ...

  • Selenium2+python自动化55-unittest之装饰器(@classmethod)

    前言 前面讲到unittest里面setUp可以在每次执行用例前执行,这样有效的减少了代码量,但是有个弊端,比如打开浏览器操作,每次执行用例时候都会重新打开,这样就会浪费很多时间. 于是就想是不是可以 ...

  • 神奇的Python property装饰器:1行代码让Python方法秒变属性

    神奇的Python property装饰器:1行代码让Python方法秒变属性