Python中装饰器的基本功能理解

目录
  • 前言
  • 什么是装饰器
  • Python 函数的基本特性
    • 函数名的本质:
    • 将函数作为变量使用:
  • 进一步实现装饰器
    • 使用Python装饰器语句:
  • 总结

前言

在 python 中,装饰器由于是 python 语言自带的一个功能,因此,对于其实现以及其用法就会感到比较奇怪,这里我记录一下对它的理解,加深自己的印象。

什么是装饰器

对于什么是装饰器,我们其实应该知道为什么会存在装饰器。

​ 装饰器是 python 引入的一个非常有意思的功能,它主要用于解决想要在原有函数或类的基础上进行功能扩展,但又不会破坏这个函数本身的功能。并且我们可以方便的添加和去除这一部分扩展的功能的需求。

​ 例如:当你在调试代码的时候,你想要给一些功能函数添加打印调试信息。但是,这个功能只在我们调试的时候使用,正式发布的时候是不需要的。如果此时,你在函数内部修改,由于有多个函数都需要添加相同的或类似的信息,那么,你就需要逐一修改,这个就有点麻烦了,此时我们就可以使用装饰器来解决这一问题。

​ 在解释如何使用装饰器之前,我们需要先把和装饰器有关的基本概念给讲一下。

Python 函数的基本特性

函数名的本质:

在 python 中,一切皆是对象,也就是说,我们定义的变量和函数都是一个对象。而是对象就意味着我们可以获得这个对象的属性,例如函数对象有一个 __name__ 的属性:

def function(): #定义一个函数
    print('this is a function !')

function()
print(function) #打印函数名的地址
print(function.__name__) #打印函数名

a = function #把函数赋给一个变量
a()
print(a) #打印 a 的地址
print(a.__name__) #再次打印函数名

​ 打印结果:

this is a function !
<function function at 0x0000029F83C17F70>
function
this is a function !
<function function at 0x0000029F83C17F70>
function

由打印可以看出,我们的函数名在赋给另一个变量的时候,其函数地址和函数属性中的函数名是没有变化的。也就是说,当我们在定义函数的时候,我们的函数名和普通的变量是一样的,唯一的不同就是,我们的函数名会指向一个内存空间,而这片空间内保存的是一个函数体的内容。

​ 但是,当我们把这个函数名赋值给其他变量的时候,这个函数名就会把它执行的内存空间的地址赋值给另一个变量,因此,另一个变量也就成为了一个函数了。

​ 这里我们已经能够注意到了,函数名如果不加 () 那么它和普通的变量一样,而加了 () 之后,它就会去执行我们的函数内容。

​ 这里我们把试着删除我们定义时候使用的函数名:

del function #删除 function 函数
a() #执行 a()
print(a) #打印出 a 指向的地址
print(a.__name__) #打印a的函数名

function()
print(function)
print(function.__name__)

​ 查看打印:

this is a function !
<function function at 0x000002258DC17F70>
function
NameError: name 'function' is not defined

​ 可以看到,我们的 function() 函数名提示没有定义,但是我们的 a() 函数却可以正常的打印出来。这里的 del 其实就是把我们的 function 这个函数名的指针给指向的函数地址给删去了,此时它变成立一个真正的未定义的变量了。

将函数作为变量使用:

​ 既然函数名和普通的变量可以相互赋值,那就说明,我们也可以像使用普通变量一样使用函数名了。

在函数中定义函数:

​ 我们可以像定义普通变量一样,在一个函数中定义另一个函数:

def function1():
    print('this is function 1')
    def function2():
        print('this is function 2 !')
        return 0
    function2()
    return 0

function1()
function2()

​ 打印如下:

this is function 1
this is function 2 !
NameError: name 'function2' is not defined

​ 可以看到,我们在 function1 中定义了一个 function2 函数,而且在 function1 中使用了 function2 这个函数。但是,当我们在外面使用 function2 这个函数的时候,却打印了该函数未定义。这里说明,函数内定义的函数的作用域也仅限于函数内部,和我们的局部变量是一样的。

​ 但是,当我们把函数作为返回值的时候,这个情况就不一样了,这里参考我上一篇文章:Python中的闭包中的变量作用域问题

在函数中返回函数名:

既然我们可以在一个函数中定义另一个函数,那么也就可以在函数中返回另一个函数:

def function1():
    print('this is function 1')
    def function2():
        print('this is function 2 !')
        return 0
    function2() #在函数内部使用该函数
    return function2 #返回该函数的函数名

a = function1() #把函数名返回给一个变量
a()

​ 打印如下:

this is function 1
this is function 2 !
this is function 2 !

​ 这里可以看到,我们的这个在函数 function1 中定义并返回了函数 function2 并在外部使用一个变量来接收 funciton1 的返回值。由此可以看出,函数名和变量的使用方式差别不大。

​ 注意: 虽然我们说的时候会说在一个函数中返回另一个函数,但是,实际上,我们返回的只是这个函数的函数名(不带括号'()‘)。

把函数名作为参数使用:

def hello():
    print("hello")

def function1(func): #接收一个参数
    print('before call hello !')
    func()
    print('after call hello !')
    #function2() #在函数内部使用该函数

function1(hello) #把 hello 作为参数传递进去

​ 打印如下:

before call hello !
hello
after call hello !

​ 由打印可以知道,我们在函数 function1 中定义的接收参数 func 我们在定义的时候并没有采用什么特殊的方式,而是和普通参数一样定义。之后,在外部调用 function1 的使用,把函数名 hello 当作参数传递进去了。随后,我们运行 function1 并在 function1 中成功调用了 hello 函数。

进一步实现装饰器

​ 现在,让我们再重新看一下什么是装饰器,我们在上面的把函数名作为参数使用时,已经实现了一个和装饰器功能类似的函数了。假如我们的 hello() 函数是我们的功能函数,而 function1 作为我们的装饰器,那么,我们成功实现了在不改变 hello() 函数的基础上,通过把它作为参数使用而增加了其他的打印内容。

​ 虽然我们上面实现了一个类似装饰器的功能,但是,我们可以看到,使用这个的时候我们需要每次都给 function1 传入一个函数,这样使用就很麻烦了。下面我们改造一下这个函数:

def decorator(func): #装饰器函数,用于接收一个函数参数
    def wrapper(): #定义一个内函数 wrapper
        print('before call hello !')
        func()
        print('after call hello !')
    return wrapper #把内函数做未返回值

def hello():
    print("hello")

hello = decorator(hello) #重新定义一个函数 hello1
hello() #执行 hello

​ 打印如下:

before call hello !
hello
after call hello !

​ 通过上面的打印可以看到,我们新更改的这个函数可以实现和上面的函数一样的功能。但是,我们这里在使用它的时候比之前要简单一些,因为我们可以直接使用旧的函数名来使用新的功能 (这里我们相当于给函数名 hello 赋值了一个新的函数wrapper),当我们想要使用旧函数时,只需要把 hello=function(hello) 这行内容给注释掉就可以了。

使用Python装饰器语句:

简单装饰器

​ 上面的我们已经实现了一个装饰器的功能,下面我们使用 Python 中自带的装饰器来测试一下:

def decorator(func): #装饰器函数,用于接收一个函数参数
    def wrapper(): #定义一个内函数 wrapper
        print('before call hello !')
        func()
        print('after call hello !')
    return wrapper #把内函数做为返回值

@decorator # '@' 是系统自带的装饰器语法糖
def hello():
    print('hello')

hello()

​ 打印如下:

before call hello !
hello
after call hello !

​ 可以看到,我们使用系统自带得装饰器语法实现了和我们上面得函数一样得功能。

​ 这里我们可以看到,他们两个唯一得不同就是使用了 @decorator这个符号来代替了 hello=decorator(hello) 这个赋值语句。

​ 到这里,其实我们基本上就已经明白了python所谓得装饰器的原理和实际用法了,但是,这里我们还有一个问题,那就是这种方法会改变我们的函数的属性吗?

​ 我们测试一下:

print(hello.__name__)

​ 打印如下:

wrapper

​ 很明显,我们的原函数的属性中的函数名被更改了,其实通过上面自己的实现,我们可以发现,我们使用装饰器语法其实就是新建了一个函数名,然后用它去接收装饰器函数的返回函数名,这样,该该函数肯定还是继承了装饰器返回函数的函数名了。

​ 为了解决这个问题,我们可以使用如下方法:

import functools 

def function(func): #接收一个参数
    @functools.wraps(func)
    def wrapper(): #定义一个内函数 wrapper
        print('before call hello !')
        func()
        print('after call hello !')
    return wrapper #把内函数做为返回值

@function # '@' 是系统自带的装饰器语法糖
def hello():
    print('hello')

hello()
print(hello.__name__)

​ 打印如下:

before call hello !
hello
after call hello !
hello

​ 通过我们使用系统模块,我们解决了这一问题。

带参数装饰器:

上面我们展示了装饰器的基础用法,但是,我们可以发现一个问题,那就是这个装饰器只能用于打印一类的基本操作,有时我们需要在装饰器函数内传参,且需要在多个函数中使用同一个装饰器函数,如果单纯使用上面的方法就不太容易操作了。

​ 下面我们展示一种给装饰器传参的操作方法:

import functools

def logging(level):#装饰器接收参数函数
    def decorator(func): #装饰器函数,用于接收一个函数
        @functools.wraps(func)
        def wrapper(*args, **kwargs): #定义一个内函数 wrapper
            if level == 'warn':
                print('warn: before call %s !' %func.__name__)
                func()
                print('warn: after call %s !' %func.__name__)
            if level == 'error':
                print('error: before call %s !' %func.__name__)
                func()
                print('error: after call %s !' %func.__name__)
        return wrapper #把内函数做为返回值
    return decorator

@logging(level='warn') # '@' 是系统自带的装饰器语法糖
def hello():
    print('hello')

@logging(level='error')
def function1():
    print('function1')

hello()
function1()
print(hello.__name__)
print(function1.__name__)

​ 打印如下:

warn: before call hello !
hello
warn: after call hello !
error: before call function1 !
function1
error: after call function1 !
hello
function1

​ 可以看到,我们在两个函数中使用了一个装饰器语法,而且给这个装饰器分别传了不同的参数,这个才比较符号我们实际可能会用到的情况。

​ 这里第一次看可能感觉有点复杂,而且我们在这里也使用了多层函数嵌套,每层都传不同的参数。这里我来仔细拆分一下这个函数:

​ 首先我们知道:

@logging
def hello():
     print('hello')
#等价于
logging(hello)

​ 因此:

@logging(level='warn')
def hello():
     print('hello')
#等价于
logging(hello)(level='warn')

​ 下面我们继续拆解 logging(hello)(level=‘warn') 这句话:

logging(hello)(level='warn')

由于
 logging(hello) 返回 decorator
于是
 logging(hello)(level='warn')
    等价于
    decorator(level='warn')

 decorator 返回 wrapper
    因此
 这里其实就到了我们最上面的简单装饰器了

​ 到这里我们就明白了我们的装饰器传参是怎么回事了。

装饰器类:​

由于我们在 python 中会经常使用类来对某一功能进行封装,这样,当我们在使用某一功能的时候就更加灵活且方便了。

​ 因此,我们的 python 也给我们提供了实现装饰器类的使用方法:

class Logging(object):
    def __init__(self, func):
        self._func = func

    def __call__(self):
        print('class: before call %s !' %self._func.__name__)
        self._func()
        print('class: after call %s !' %self._func.__name__)

@Logging
def hello():
    print('Hello')

hello()

​ 打印如下:

class: before call hello !
Hello
class: after call hello !

​ 可以看到,我们的类装饰器的用法和函数类似,只是在定义装饰器函数的时候,把函数的实现变成了类方法的实现方式。

​ 除了这种最基本的的使用方式,我们其实也可以给类装饰器传参:

class Logging(object):
    def __init__(self, level='INFO'):
        self._level = level

    def __call__(self, func):
        def wrapper(*args, **kwargs):
            if self._level == 'WARN':
                print('class: warn before call %s !' %func.__name__)
                func()
                print('class: warn after call %s !' %func.__name__)
        return wrapper

@Logging(level='WARN')
def hello():
    print('Hello')

hello()

​ 打印如下:

class: warn before call hello !
Hello
class: warn after call hello !

​ 这里传参方式和上面直接在类中的 __call__ 中定义函数有些不一样,这里需要记住两点:

__init__:不再接收被装饰函数,而是接收传入参数;
__call__:接收被装饰函数,实现装饰逻辑

​ 这里就不对这个类方法进行深入解析了。

总结

到此这篇关于Python中装饰器的基本功能的文章就介绍到这了,更多相关Python装饰器功能内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • Python中的装饰器用法详解

    本文实例讲述了Python中的装饰器用法.分享给大家供大家参考.具体分析如下: 这里还是先由stackoverflow上面的一个问题引起吧,如果使用如下的代码: 复制代码 代码如下: @makebold @makeitalic def say():    return "Hello" 打印出如下的输出: <b><i>Hello<i></b> 你会怎么做?最后给出的答案是: 复制代码 代码如下: def makebold(fn):    

  • 深入理解python中的闭包和装饰器

    python中的闭包从表现形式上定义(解释)为:如果在一个内部函数里,对在外部作用域(但不是在全局作用域)的变量进行引用,那么内部函数就被认为是闭包(closure). 以下说明主要针对 python2.7,其他版本可能存在差异. 也许直接看定义并不太能明白,下面我们先来看一下什么叫做内部函数: def wai_hanshu(canshu_1): def nei_hanshu(canshu_2): # 我在函数内部有定义了一个函数 return canshu_1*canshu_2 return

  • 介绍Python的@property装饰器的用法

    在绑定属性时,如果我们直接把属性暴露出去,虽然写起来很简单,但是,没办法检查参数,导致可以把成绩随便改: s = Student() s.score = 9999 这显然不合逻辑.为了限制score的范围,可以通过一个set_score()方法来设置成绩,再通过一个get_score()来获取成绩,这样,在set_score()方法里,就可以检查参数: class Student(object): def get_score(self): return self._score def set_s

  • Python装饰器的函数式编程详解

    Python的装饰器的英文名叫Decorator,当你看到这个英文名的时候,你可能会把其跟Design Pattern里的Decorator搞混了,其实这是完全不同的两个东西.虽然好像,他们要干的事都很相似--都是想要对一个已有的模块做一些"修饰工作",所谓修饰工作就是想给现有的模块加上一些小装饰(一些小功能,这些小功能可能好多模块都会用到),但又不让这个小装饰(小功能)侵入到原有的模块中的代码里去.但是OO的Decorator简直就是一场恶梦,不信你就去看看wikipedia上的词条

  • 详解Python中的装饰器、闭包和functools的教程

    装饰器(Decorators) 装饰器是这样一种设计模式:如果一个类希望添加其他类的一些功能,而不希望通过继承或是直接修改源代码实现,那么可以使用装饰器模式.简单来说Python中的装饰器就是指某些函数或其他可调用对象,以函数或类作为可选输入参数,然后返回函数或类的形式.通过这个在Python2.6版本中被新加入的特性可以用来实现装饰器设计模式. 顺便提一句,在继续阅读之前,如果你对Python中的闭包(Closure)概念不清楚,请查看本文结尾后的附录,如果没有闭包的相关概念,很难恰当的理解P

  • Python装饰器限制函数运行时间超时则退出执行

    实际项目中会涉及到需要对有些函数的响应时间做一些限制,如果超时就退出函数的执行,停止等待. 可以利用python中的装饰器实现对函数执行时间的控制. python装饰器简单来说可以在不改变某个函数内部实现和原来调用方式的前提下对该函数增加一些附件的功能,提供了对该函数功能的扩展. 方法一. 使用 signal # coding=utf-8 import signal import time def set_timeout(num, callback): def wrap(func): def h

  • python中多个装饰器的执行顺序详解

    装饰器是程序开发中经常会用到的一个功能,也是python语言开发的基础知识,如果能够在程序中合理的使用装饰器,不仅可以提高开发效率,而且可以让写的代码看上去显的高大上^_^ 使用场景 可以用到装饰器的地方有很多,简单的举例如以下场景 引入日志 函数执行时间统计 执行函数前预备处理 执行函数后清理功能 权限校验等场景 缓存 今天讲一下python中装饰器的执行顺序,以两个装饰器为例. 装饰器代码如下: def wrapper_out1(func): print('--out11--') def i

  • python装饰器简介---这一篇也许就够了(推荐)

    Python装饰器(decorator)是在程序开发中经常使用到的功能,合理使用装饰器,能让我们的程序如虎添翼. 装饰器引入 初期及问题诞生 假如现在在一个公司,有A B C三个业务部门,还有S一个基础服务部门,目前呢,S部门提供了两个函数,供其他部门调用,函数如下: def f1(): print('f1 called') def f2(): print('f2 called') 在初期,其他部门这样调用是没有问题的,随着公司业务的发展,现在S部门需要对函数调用假如权限验证,如果有权限的话,才

  • python 一篇文章搞懂装饰器所有用法(建议收藏)

    01. 装饰器语法糖 如果你接触 Python 有一段时间了的话,想必你对 @ 符号一定不陌生了,没错 @ 符号就是装饰器的语法糖. 它放在一个函数开始定义的地方,它就像一顶帽子一样戴在这个函数的头上.和这个函数绑定在一起.在我们调用这个函数的时候,第一件事并不是执行这个函数,而是将这个函数做为参数传入它头顶上这顶帽子,这顶帽子我们称之为装饰函数 或 装饰器. 你要问我装饰器可以实现什么功能?我只能说你的脑洞有多大,装饰器就有多强大. 装饰器的使用方法很固定: 先定义一个装饰函数(帽子)(也可以

  • Python中的各种装饰器详解

    Python装饰器,分两部分,一是装饰器本身的定义,一是被装饰器对象的定义. 一.函数式装饰器:装饰器本身是一个函数. 1.装饰函数:被装饰对象是一个函数 [1]装饰器无参数: a.被装饰对象无参数: 复制代码 代码如下: >>> def test(func):     def _test():         print 'Call the function %s().'%func.func_name         return func()     return _test >

随机推荐