当前位置: 首页 > news >正文

python中的装饰器(基础装饰器)

文章目录

    • 一 前置知识-高阶函数,闭包
        • 1. 高阶函数
        • 2. 闭包
    • 二 函数装饰器
        • 1. 什么是装饰器(原理)?
        • 2. 装饰器的实现
        • 3. 何时执行装饰器
        • 4. wraps方法
    • 三 类装饰器

一 前置知识-高阶函数,闭包

1. 高阶函数

在python中,如果一个函数的参数是另外一个或几个函数,那么这个函数就是高阶函数,如下

#高阶函数
def fun1():
    print("Hello world")

def fun2(fun):
    print("start".center(20, '='))
    fun()
    print("end".center(20, '='))

fun2(fun1)

>>>
=======start========
Hello world
========end=========

上面的函数fun2就是一个高阶函数,因为它的参数是一个函数fun1。

2. 闭包

在python中,闭包是一个函数,它延伸了变量的作用域,使得在定义变量的作用域失效后,该变量仍然能够被调用
了解闭包更有助于学习装饰器,关于闭包,可以参考这篇文章:python之闭包


二 函数装饰器

1. 什么是装饰器(原理)?

装饰器,顾名思义就是装饰XXX的工具。在python中,装饰器的本质就是一个高阶函数,它接受一个函数作为参数,并返回一个被装饰后的函数。
装饰器的作用如下

  • 在不修改被装饰函数的源代码和调用方式的情况下,给被装饰函数添加额外的功能。

即就是你传一个函数给装饰器,装饰器不会改变该函数的代码和调用方式就能使该函数获得额外的功能。

2. 装饰器的实现

比如要实现一个计算函数运行时间的功能,该怎么实现呢?
首先你可以使用高阶函数这样写

#装饰器
import time

def fun():
    time.sleep(2)

def timer(fun):
    start_time = time.time()
    fun()
    end_time = time.time()
    total = end_time - start_time
    print("函数运行时间为:{}".format(total))

timer(fun)

>>>
函数运行时间为:2.000129222869873

上面这中写法可以实现计算函数的运行时间,但是有个缺点,就是每计算一个函数运行的时间就得调用一次timer函数,如果函数有几十个几百个,那么就得调用几十次几百次time函数,而且也不太直观。

下面再来用 闭包 优化一下,如下

#装饰器
import time

def fun():
    time.sleep(2)

def timer(fun):

    def wrapper():
        start_time = time.time()
        fun()
        end_time = time.time()
        print("函数运行时间为:{}".format(end_time-start_time))

    return wrapper

#timer返回嵌套函数的引用wrapper
#fun=wrapper
fun = timer(fun)
#调用fun()就是调用wrapper()
fun()

>>>
函数运行时间为:2.0008482933044434

上面用闭包函数优化了一下,现在计算函数运行时间的功能由wrapper函数来实现,而wrapper函数是嵌套在timer函数里面,timer函数返回wrapper()函数的引用wrapperfun=timer(fun)就相当于fun=wrapper,调用fun() 就相当于调用 wrapper()
可以看到, 经过闭包优化后,我们的调用方式变了,不再是调用 timer() 了,而是直接调用函数本身 fun() 就可以计算函数的运行时间了。

其实上面用闭包优化了后的 timer() 函数就是一个函数装饰器,因为它既没有修改被装饰函数fun() 的代码,也没有修改器调用方式就给函数 fun() 实现了额外计算运行时间的功能。

python又用 @ 来代替fun=timer(fun), @timer和fun=timer(fun)是等效的,于是上面的调用就可以写成下面的形式

import time

def timer(fun):

    def wrapper():
        start_time = time.time()
        fun()
        end_time = time.time()
        print("函数运行时间为:{}".format(end_time-start_time))

    return wrapper

#@timer和fun=timer(fun)等效
@timer
def fun():
    time.sleep(2)

fun()

>>>
函数运行时间为:2.0007741451263428

可以看到,用 @timer 替换fun=timer(fun) 后我们就可以直接编写函数,然后调用函数就行了,这就是装饰器的魅力所在!

3. 何时执行装饰器

在python中,装饰器还有下面这两个特性:

  • 被装饰函数和装饰器在同一个模块时,只有在明确调用被装饰函数时装饰器才被执行
  • 当被装饰函数和装饰器在不同的模块时,只要被装饰函数一经定义,装饰器就会立即执行,这一般在import导入时发生。

被装饰函数和装饰器在同一个模块时

#装饰器
import time

def timer(fun):
    print("我是老六")

    def wrapper():
        start_time = time.time()
        fun()
        end_time = time.time()
        print("函数运行时间为:{}".format(end_time-start_time))

    return wrapper

#@timer和fun=timer(fun)等效
@timer
def fun():
    time.sleep(2)

if __name__ == '__main__':
    fun()

>>>
我是老六
函数运行时间为:2.000951051712036

被装饰函数和装饰器不在同一个模块

import time
from CSDN import timer

@timer
def fun1():
    time.sleep(1)

fun1()

>>>
我是老六
我是老六
函数运行时间为:1.000683069229126

可以看到,结果打印了两次“我是老六”,这是因为在import时装饰器就执行了一次,在调用被装饰函数fun1时,装饰器又执行了一次。

4. wraps方法

在上面我们用闭包来优化写装饰器时,说过timer()函数返回的是wrapper,而我们在调用装饰器时是这样的

#@timer和fun=timer(fun)等效
@timer
def fun():
    time.sleep(2)

fun()

说明在调用fun() 时其实是调用的 wrapper(),这时候 fun() 的__ name __ 属性已经被改变了,如下

#@timer和fun=timer(fun)等效
@timer
def fun():
    time.sleep(2)

print(fun.__name__)

>>>
wrapper

可以看到fun()的__ name __ 属性已经被改成了wrapper,我们当然不希望装饰器改变被装饰函数的任何属性,这时我们就可以用functools模块中的wraps方法还原被装饰函数的__name__属性,如下

#装饰器
import time
import functools

def timer(fun):
    @functools.wraps(fun)
    def wrapper():
        start_time = time.time()
        fun()
        end_time = time.time()
        print("函数运行时间为:{}".format(end_time-start_time))

    return wrapper

#@timer和fun=timer(fun)等效
@timer
def fun():
    time.sleep(2)

print(fun.__name__)

>>>
fun

三 类装饰器

上面讲的是函数装饰器,下面简单介绍下类装饰器。把上面实现计算函数运行时间的功能用类装饰实现,如下

#类装饰器
import time

class timer():

    def __init__(self, func):
        self.func = func

    def __call__(self):
        star_time = time.time()
        self.func()
        end_time = time.time()
        total = end_time - star_time
        print("函数运行时间:", total)

@timer
def fun():
    time.sleep(2)

fun()

>>>
函数运行时间: 2.0005552768707275

可以看到类装饰器实现的效果和函数装饰器实现的效果是一样的,只不过类装饰器在内部的装饰函数是用__call__ 方法实现的。其中 __ call __ 的作用如下:

  • 能够使类的实例像函数调用那样被调用
@timer等价于 fun=timer(fun)

@timer 的在这里的作用实际就是实例化一个fun对象(fun=timer(fun)),对于类的实例化对象是不支持** 实例化对象() **这样调用的,而__call__ 方法的作用就是支持实例化对象这样被调用。所以才满足装饰器不改变被装饰函数调用方式的特性。

以上就是装饰器相关的一些基础知识。

相关文章:

  • 多级缓存与局部性原理
  • NLP冻手之路(2)——文本数据集的下载与各种操作(Datasets)
  • 【Java】异常
  • 第22章 软件安装 RPM/ YUM
  • 手把手教你如何使用YOLOV5训练自己的数据集
  • 公众号网课搜题接口
  • 八、创建JWT工具类
  • 这家公司只有1个人,年赚一个亿
  • 【前端攻城师之JS基础】02JS对象基础
  • 【C/C++】程序环境,探索程序的执行过程(习得无上内功《易筋经》的第一步)
  • 计算机网络——基于UDP与TCP网络编程
  • 中兴设备18种命令模式总结大全,全网第一篇,强烈建议收藏!
  • 【学生管理系统】权限管理
  • NLP学习之:Bert 模型复现(1)任务分析 + 训练数据集构造
  • 【PTA】《数据结构与算法》线性结构复习题
  • [nginx文档翻译系列] 控制nginx
  • 30天自制操作系统-2
  • Druid 在有赞的实践
  • ERLANG 网工修炼笔记 ---- UDP
  • ESLint简单操作
  • Linux编程学习笔记 | Linux多线程学习[2] - 线程的同步
  • python大佬养成计划----difflib模块
  • Vue小说阅读器(仿追书神器)
  • 翻译--Thinking in React
  • 吐槽Javascript系列二:数组中的splice和slice方法
  • 我的zsh配置, 2019最新方案
  • 我是如何设计 Upload 上传组件的
  • 写代码的正确姿势
  • 原创:新手布局福音!微信小程序使用flex的一些基础样式属性(一)
  • - 转 Ext2.0 form使用实例
  • Semaphore
  • ​sqlite3 --- SQLite 数据库 DB-API 2.0 接口模块​
  • #【QT 5 调试软件后,发布相关:软件生成exe文件 + 文件打包】
  • ( 10 )MySQL中的外键
  • (1)(1.13) SiK无线电高级配置(五)
  • (1)SpringCloud 整合Python
  • (51单片机)第五章-A/D和D/A工作原理-A/D
  • (ctrl.obj) : error LNK2038: 检测到“RuntimeLibrary”的不匹配项: 值“MDd_DynamicDebug”不匹配值“
  • (Ruby)Ubuntu12.04安装Rails环境
  • (安全基本功)磁盘MBR,分区表,活动分区,引导扇区。。。详解与区别
  • (八)五种元启发算法(DBO、LO、SWO、COA、LSO、KOA、GRO)求解无人机路径规划MATLAB
  • (三) diretfbrc详解
  • (五)IO流之ByteArrayInput/OutputStream
  • (一)appium-desktop定位元素原理
  • (转)chrome浏览器收藏夹(书签)的导出与导入
  • .babyk勒索病毒解析:恶意更新如何威胁您的数据安全
  • .equal()和==的区别 怎样判断字符串为空问题: Illegal invoke-super to void nio.file.AccessDeniedException
  • .net mvc actionresult 返回字符串_.NET架构师知识普及
  • .NET 动态调用WebService + WSE + UsernameToken
  • /etc/fstab和/etc/mtab的区别
  • ::before和::after 常见的用法
  • [ 代码审计篇 ] 代码审计案例详解(一) SQL注入代码审计案例
  • [20180312]进程管理其中的SQL Server进程占用内存远远大于SQL server内部统计出来的内存...
  • [Android Studio 权威教程]断点调试和高级调试
  • [Bada开发]初步入口函数介绍