装饰器是一个用于封装函数或类的代码的工具。它显式地将封装器应用到函数或类上,从而使它们选择加入到装饰器的功能中。对于在函数运行前处理常见前置条件(例如确认授权),或在函数运行后确保清理(例如输出清除或异常处理)装饰器都非常有用。对于处理已经被装饰的函数或类本身,装饰器也很有用。例如,装饰器可以将函数注册到信号系统,或者注册到Web应用程序的URI注册表中。

    本文将概要介绍什么是装饰器,以及装饰器如何与 Python的函数和类交互。

1.1理解装饰器

    究其核心而言,装饰器就是一个可以接受调用也可以返回调用的调用。装饰器无非就是一个函数(或调用,如有_call_method_方法的对象),该函数接受被装饰的函数作为其位置参数。装饰器通过使用该参数来执行某些操作,然后返回原始参数或一些其他的调用(大概以这种方式与装饰器交互)由于函数在 Python中是一级对象,因此它们能够像其他对象一样被传递到另一个函数。装饰器就是接受另一个函数作为参数,并用其完成一些操作的函数。实际上这很容易理解。考虑下面一个非常简单的装饰器。它仅仅为被装饰的调用点字符串附加了一个字符串。
 def decorated_by(func):
 
   func. __doc__ +="\ Decorated by decorated_by"
   return func

现在,考虑下面这个普通的函数:
def add(x, y ):
  "'Return the sum of x and y'"
  return x +y
 
函数的 docstring是在第一行指定的字符串。假如在 Python的 Shell中对该函数运行Help命令,就能看到该字符串。下面是将装饰器应用到add函数的示例
def add(x, y): 
   "'return the sum of x and y "'
   return x +y
add=decorated_by(add)
 
以下是执行help命令得到的结果
Help on function add in module__main__:
add(x, y)
  Return the sum of x and y.
  Decorated by decorated_by
(END)
 这里发生了什么?其实就是装饰器修改了add函数的__doc__属性,然后返回原来的函数对象。
1.2装饰器语法

    大多数时候开发人员使用装饰器来装饰函数,他们只对装饰过的最终函数感兴趣,而对于未装饰函数的引用最终就变得多余,正因为如此(也是为了更整洁),所以定义函数,给它赋一个特定的名称,然后立刻将装饰过的函数赋给相同的名称就不可取.因此, Python为装饰器引入了特殊的语法。装饰器的应用是通过在装饰器的名称前放置一个@字符,并在被装饰函数声明之上添加一行(不包含隐式装饰器的方法签名)来实现的。

下面来看一下如何优先将 decorated_by装饰器应用到add方法:
@decorated_by
def add(x,y):
 "'Return the sum of x +y"'
   return x+y

     再次注意,这里没有给@ decorated_by提供方法签名。假设该装饰器有一个单独的位置参数,而这个参数是装饰过的方法(在某些情况下,会看到一个带有其他参数的方法签名)
该语法允许在声明函数的位置应用装饰器,从而代码更容易阅读并且可以立即意识到应用了装饰器。可读性很重要。
 

装饰器应用的顺序:

    何时应用装饰器?使用@语法时,在创建被装饰的可调用函数后,会立刻应用装饰器。因此以上两个示例中所展示的将 decorated_by应用到ad的方式几乎是一样的。首先创建add函数,然后立即使用 decorated_by将其封装起来。
    需要注意的重要一点是,对某个可调用函数,可以使用多个装饰器(就像可以多次封装函数调用一样)。
    但请注意,如果通过@语法使用多个装饰器,就需要按照自底向上的顺序来应用它们。起初觉得这违反直觉,但是恰恰说明了 Python 解释器实际所做的工作。
    考虑下面这个应用了两个装饰器的函数:
@also_decorated_by
@decorated_by
def add(x,y):
   "'Return the sum of x and y. ""
   return x+y
 
    首先发生的是由解释器创建add函数,然后应用 decorated_by装饰器。该装饰器返回了一个可调用函数(正如所有装饰器做的一样),该函数被发送给also_ decorated_by装饰器,also_decorated_by也做了同样的事情,接下来结果被赋给add函数。
    切记,装饰器 decorated_by的应用程序与下面的代码在语法上是相同的:
 

add=decorated_by(add)

    前面两个装饰器示例与下面的代码在语法上相同
 

add =also_decorated_by(decorated_by(add))

 
    在这两种情况下,读取代码时首先读到装饰器also_decorated_by。但是,装饰器的应用是自底向上的,这与函数的解析(由内向外)是相同的。工作也采用同样的原则。

    在传统的函数调用情况下,解释器一定首先解析内部函数调用,以便有合适的对象或值发送给外部调用。
       add=also_decorated_by(decorated_by(add)) #First, get a return value for
                                          #decorated_by(add)
        add = also decorated_by(decorated_by(add))#Send that return value to
                                           #also decorated b
    
     有了装饰器后,通常首先创建add函数。

    @also_decorated_by
    @decorated_by
    def add(x, y):
       "'teturn the sum sf x and Y."'
       return x+y
 
    然后,调用装饰器@ decorated_by,并将其作为add函数的装饰方法。
    @also_decorated_by
    @decorated_by
    def add(x, y):
        "'Return the sum of x and y."'
        return x+y
 
    @decorated_by函数返回自己的可调用函数(在本例中,是add的修改版本)。该返回值在最后步骤发送给@also_decorated_by。
    @also_decorated_by
    @decorated_by
    def add(x, y):
        "'Return the sum of x and y."'
        return x+y
    应用装饰器时需要记住一件重要的事情,即装饰器的应用是自底向上的。很多时候,顺序非常重要。

1.3在何处使用装饰器
    Python标准库中包括很多包含装饰器的模块,并且很多常用工具和框架利用它们实现常用功能。
    例如,如果要使一个类上的方法不需要这个类的实例,可以使用@classmethod或@staticmethod装饰器,它们是标准库的一部分。mock模块(用于单元测试,在 Python3.3以后被添加到标准库中)允许使用@mock. patch或@mock. patch. object作为装饰器。
    一些常见工具也使用装饰器。 Django(用于 Python的常见Web框架)使用@login_required作为装饰器,允许开发人员指定用户必须登录オ能査看一个特定页面,并且使用@ permission_ required应用更具体的权限限制。Fask(另一个常见的Web框架)使用@app. route充当指定的URI与浏览器访问这些URI时所运行的函数之间的注册表。
    Celery(常见的 Python任务运行工具)使用复杂的@task装饰器来标识函数是否为异步任务。该装饰器实际上返回Task类的实例,用来阐明如何使用装饰器制作一个方便的APl;

python语言中的装饰器详解的更多相关文章

  1. python设计模式之装饰器详解(三)

    python的装饰器使用是python语言一个非常重要的部分,装饰器是程序设计模式中装饰模式的具体化,python提供了特殊的语法糖可以非常方便的实现装饰模式. 系列文章 python设计模式之单例模 ...

  2. C#中的 Attribute 与 Python/TypeScript 中的装饰器是同个东西吗

    前言 最近成功把「前端带师」带入C#的坑(实际是前端带师开始从cocos转unity游戏开发了) 某天,「前端带师」看到这段代码后问了个问题:[这个是装饰器]? [HttpGet] public Re ...

  3. Python装饰器详解

    python中的装饰器是一个用得非常多的东西,我们可以把一些特定的方法.通用的方法写成一个个装饰器,这就为调用这些方法提供一个非常大的便利,如此提高我们代码的可读性以及简洁性,以及可扩展性. 在学习p ...

  4. python之装饰器详解

    这几天翻看python语法,看到装饰器这里着实卡了一阵,最初认为也就是个函数指针的用法,但仔细研究后发现,不止这么简单. 首先很多资料将装饰器定义为AOP的范畴,也就是Aspect Oriented ...

  5. Python中的各种装饰器详解

    Python装饰器,分两部分,一是装饰器本身的定义,一是被装饰器对象的定义. 一.函数式装饰器:装饰器本身是一个函数. 1.装饰函数:被装饰对象是一个函数 [1]装饰器无参数: a.被装饰对象无参数: ...

  6. Python—装饰器详解

    装饰器:(语法糖) 本质是函数,它是赋予函数新功能,但是不改变函数的源代码及调用方式   原则: 1.不能修改被装饰函数的源代码 2.不能修改被装饰函数的调用方式 3.函数的返回值也不变 这两点简而言 ...

  7. python装饰器1:函数装饰器详解

    装饰器1:函数装饰器 装饰器2:类装饰器 装饰器3:进阶 先混个眼熟 谁可以作为装饰器(可以将谁编写成装饰器): 函数 方法 实现了__call__的可调用类 装饰器可以去装饰谁(谁可以被装饰): 函 ...

  8. Python全栈开发之8、装饰器详解

    一文让你彻底明白Python装饰器原理,从此面试工作再也不怕了.转载请注明出处http://www.cnblogs.com/Wxtrkbc/p/5486253.html 一.装饰器 装饰器可以使函数执 ...

  9. python函数装饰器详解

    python装饰器(fuctional decorators)简单来说就是修改其他函数的函数. 这样的函数需要满足两个个条件: 1.不能修改原函数的源代码 2.不能改变原函数的调用方式 需要达到的效果 ...

  10. python 叠加装饰器详解

    def out1(func1): #7.func1=in2的内存地址,就是in2 print('out1') def in1(): #8.调用函数index() 因为函数在in1里,所以首先运行in1 ...

随机推荐

  1. 微信小程序获取手机号流程

    小程序中获取手机号前提 小程序需企业认证,才可以获取用户的手机号,个人开发者是不能获取的 哔哔下 官方文档给出需先登录才可获取手机号 传送门 思路为:login登录获取code-->code传给 ...

  2. Oracle 获取月初、月末时间,获取上一月月初、月末时间

    select trunc(sysdate, 'mm'), last_day(trunc(sysdate)), trunc(add_months(trunc(sysdate), -1), 'mm'), ...

  3. QT实现参数批量配置

    QT实现批量配置 需求 一些参数需要批量化配置 之前搭建的FPGA的寄存器控制模型 使用AXI-lite搭建 直接操作上位机 这里需要一个可以快速配置所有参数的上位机 需要保存文件,可以保留上一次的参 ...

  4. FPGA 原语之一位全加器

    FPGA原语之一位全加器 1.实验原理 一位全加器,三个输入,两个输出.进位输出Cout=AB+BC+CA,本位输出S=A异或B异或C.实验中采用三个与门.一个三输入或门(另外一个是两个或门,功能一致 ...

  5. PLC:自动纠正数据集噪声,来洗洗数据集吧 | ICLR 2021 Spotlight

     论文提出了更通用的特征相关噪声类别PMD,基于此类噪声构建了数据校准策略PLC来帮助模型更好地收敛,在生成数据集和真实数据集上的实验证明了其算法的有效性.论文提出的方案理论证明完备,应用起来十分简单 ...

  6. 初学STM32 CAN通信(一)

    # 初学STM32 CAN通信(一) 1. CAN协议简介 ​ CAN是控制器局域网络(Controller Area Network)的简称, 是国际上应用最广泛的现场总线之一 ,近年来,它具有的高 ...

  7. 13 JavaScript关于prototype(超重点)

    13 JavaScript关于prototype(超重点) prototype是js里面给类增加功能扩展的一种模式. 写个面向对象来看看. function People(name, age){ th ...

  8. 2023 LGR 非专业级别软件能力认证第一轮(初赛)S组

    计算器.背包.代码都不能带进考场 禁赛三年并全国通报 B选项符合while语句 弱类型编程语言指的是可以进行类型转换,可以参与各种类型变量的运算 \[3\times 60(秒)\times 44.1\ ...

  9. #树链剖分,线段树#洛谷 2146 [NOI2015]软件包管理器

    题目传送门 分析 安装时1到\(x\)路径上都变为1,删除时\(x\)的子树都变为0, 显然可以用树链剖分+线段树实现 代码 #include <cstdio> #include < ...

  10. JDK12的新特性:teeing collectors

    目录 简介 talk is cheap, show me the code Teeing方法深度剖析 Characteristics 总结 简介 JDK12为java.util.stream.Coll ...