返回顶部
首页 > 资讯 > 后端开发 > Python >Python中的装饰器知识点有哪些
  • 801
分享到

Python中的装饰器知识点有哪些

2023-07-02 10:07:17 801人浏览 八月长安

Python 官方文档:入门教程 => 点击学习

摘要

这篇文章主要介绍“python中的装饰器知识点有哪些”,在日常操作中,相信很多人在Python中的装饰器知识点有哪些问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Python中的装饰器知识点有哪些”的疑惑有所

这篇文章主要介绍“python中的装饰器知识点有哪些”,在日常操作中,相信很多人在Python中的装饰器知识点有哪些问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Python中的装饰器知识点有哪些”的疑惑有所帮助!接下来,请跟着小编一起来学习吧!

Python中的装饰器知识点有哪些

一、闭包

要了解什么是装饰器(decorator),我们首先需要知道闭包(closure)的概念。

闭包,又称闭包函数或者闭合函数,通俗一点来讲,当某个函数被当成对象返回时还夹带了外部变量,就形成了一个闭包。

以打印Hello World为例,我们先来看一下嵌套函数的结构应该是什么样的:

def print_msg(msg):    def printer():        print(msg)    printer()print_msg('Hello World')# Hello World

执行 print_msg('Hello World') 相当于执行了 printer(),也就是执行 print(msg),所以将输出 Hello World

我们再来看一下如果是闭包,该是什么样的结构:

def print_msg(msg):    def printer():        print(msg)    return printermy_msg = print_msg('Hello World')my_msg()# Hello World

本例中的 printer 函数就是闭包。

执行 print_msg('Hello World') 实际上是返回了如下这样一个函数,它夹带了外部变量 'Hello World'

def printer():    print('Hello World')

于是调用 my_msg 就相当于执行 printer()


那么如何判断一个函数是否是闭包函数呢?闭包函数的 __closure__ 属性里面定义了一个元组用于存放所有的cell对象,每个cell对象保存了这个闭包中所有的外部变量。而普通函数的 __closure__ 属性为 None

def outer(content):    def inner():        print(content)    return innerprint(outer.__closure__)    # Noneinner = outer('Hello World')print(inner.__closure__)    # (<cell at 0x0000023FB1FD0B80: str object at 0x0000023FB1DC84F0>,)

由此可见 outer 函数不是闭包,而 inner 函数是闭包。

我们还可以查看闭包所携带的外部变量:

print(inner.__closure__[0].cell_contents)# Hello World

说了那么多,那么闭包究竟有什么用呢?闭包存在的意义就是它夹带了外部变量(私货),如果它不夹带私货,那么就和普通的函数没有任何区别。

闭包的优点如下:

  • 局部变量无法共享和长久的保存,而全局变量可能造成变量污染,闭包既可以长久的保存变量又不会造成全局污染。

  • 闭包使得函数内局部变量的值始终保持在内存中,不会在外部函数调用后被自动清除。

二、装饰器

我们先考虑这样一个场景,假设先前编写的一个函数已经实现了4个功能,为简便起见,我们用 print 语句来代表每一个具体的功能:

def module():    print('功能1')    print('功能2')    print('功能3')    print('功能4')

现在,由于某种原因,你需要为 module 这个函数新增一个 功能5,你完全可以这样修改:

def module():    print('功能1')    print('功能2')    print('功能3')    print('功能4')    print('功能5')

但在现实业务中,直接做出这样的修改往往是比较危险的(会变得不易于维护)。那么如何在不修改原函数的基础上去为它新添一个功能呢?

你可能已经想到了使用之前的闭包知识:

def func_5(original_module):    def wrapper():        original_module()        print('功能5')    return wrapper

func_5 代表该函数主要用于实现 功能5,我们接下来将 module 传入进去来观察效果:

new_module = func_5(module)new_module()# 功能1# 功能2# 功能3# 功能4# 功能5

可以看出,我们的新模块:new_module 已经实现了 功能5

在上面的例子中,函数 func_5 就是一个装饰器,它装饰了原来的模块(为它新添了一个功能)。

当然,Python有更简洁的写法(称之为语法糖),我们可以将@符号与装饰器函数的名称一起使用,并将其放置在要装饰的函数的定义上方:

def func_5(original_module):    def wrapper():        original_module()        print('功能5')    return wrapper@func_5def module():    print('功能1')    print('功能2')    print('功能3')    print('功能4')module()# 功能1# 功能2# 功能3# 功能4# 功能5

基于此,我们可以在不修改原函数的基础上完成计时任务(计算原函数的运行时间),如下:

def timer(func):    def wrapper():        import time        tic = time.time()        func()        toc = time.time()        print('程序用时: {}s'.fORMat(toc - tic))    return wrapper@timerdef make_list():    return [i * i for i in range(10**7)]my_list = make_list()# 程序用时: 0.8369960784912109s

事实上,my_list 并不是列表,直接打印会显示 None,这是因为我们的 wrapper 函数没有设置返回值。如果需要获得 make_list 的返回值,可以这样修改 wrapper 函数:

def wrapper():    import time    tic = time.time()    a = func()    toc = time.time()    print('程序用时: {}s'.format(toc - tic))    return a

三、使用多个装饰器

假如我们要为 module 新添 功能5功能6(按数字顺序),那该如何做呢?

好在Python允许同时使用多个装饰器:

def func_5(original_module):    def wrapper():        original_module()        print('功能5')    return wrapperdef func_6(original_module):    def wrapper():        original_module()        print('功能6')    return wrapper@func_6@func_5def module():    print('功能1')    print('功能2')    print('功能3')    print('功能4')module()# 功能1# 功能2# 功能3# 功能4# 功能5# 功能6

上述过程实际上等价于:

def module():    print('功能1')    print('功能2')    print('功能3')    print('功能4')new_module = func_6(func_5(module))new_module()

此外,需要注意的是,在使用多个装饰器时,最靠近函数定义的装饰器会最先装饰该函数,如果我们改变装饰顺序,则输出结果也将改变:

@func_5@func_6def module():    print('功能1')    print('功能2')    print('功能3')    print('功能4')module()# 功能1# 功能2# 功能3# 功能4# 功能6# 功能5

四、被装饰的函数带有参数

如果被装饰的函数带有参数,那该如何去构造装饰器呢?

考虑这样一个函数:

def pide(a, b):    return a / b

当b=0 时会出现 ZeropisionError。如何在避免修改该函数的基础上给出一个更加人性化的提醒呢?

因为我们的 pide 函数接收两个参数,所以我们的 wrapper 函数也应当接收两个参数:

def smart_pide(func):    def wrapper(a, b):        if b == 0:            return '被除数不能为0!'        else:            return func(a, b)    return wrapper

使用该装饰器进行装饰:

@smart_pidedef pide(a, b):    return a / bprint(pide(3, 0))# 被除数不能为0!print(pide(3, 1))# 3.0

如果不知道要被装饰的函数有多少个参数,我们可以使用下面更为通用的模板:

def decorator(func):    def wrapper(*args, **kwargs):        # ...        res = func(*args, **kwargs)        # ...        return res  # 也可以不return    return wrapper

五、带参数的装饰器

我们之前提到的装饰器都没有带参数,即语法糖 @decorator 中没有参数,那么该如何写一个带参数的装饰器呢?

前面实现的装饰器都是两层嵌套函数,而带参数的装饰器是一个三层嵌套函数。

考虑这样一个场景。假如我们在为 module 添加新功能时,希望能够加上实现该功能的开发人员的花名,则可以这样构造装饰器(以 功能5 为例):

def func_5_with_name(name=None):    def func_5(original_module):        def wrapper():            original_module()            print('功能5由{}实现'.format(name))        return wrapper    return func_5

效果如下:

@func_5_with_name(name='若水')def module():    print('功能1')    print('功能2')    print('功能3')    print('功能4')module()# 功能1# 功能2# 功能3# 功能4# 功能5由若水实现

对于这种三层嵌套函数,我们可以这样理解:当为 func_5_with_name 指定了参数后,func_5_with_name(name='若水') 实际上返回了一个 decorator,于是 @func_5_with_name(name='若水') 就相当于 @decorator

六、使用类作为装饰器

将类作为装饰器,我们需要实现 __init__ 方法和 __call__ 方法。

以计时器为例,具体实现如下:

class Timer:    def __init__(self, func):        self.func = func    def __call__(self):        import time        tic = time.time()        self.func()        toc = time.time()        print('用时: {}s'.format(toc - tic))@Timerdef make_list():    return [i**2 for i in range(10**7)]make_list()# 用时: 2.928966999053955s

如果想要自定义生成列表的长度并获得列表(即被装饰的函数带有参数情形),我们就需要在 __call__ 方法中传入相应的参数,具体如下:

class Timer:    def __init__(self, func):        self.func = func    def __call__(self, num):        import time        tic = time.time()        res = self.func(num)        toc = time.time()        print('用时: {}s'.format(toc - tic))        return res@Timerdef make_list(num):    return [i**2 for i in range(num)]my_list = make_list(10**7)# 用时: 2.8219943046569824sprint(len(my_list))# 10000000

如果要构建带参数的类装饰器,则不能把 func 传入 __init__ 中,而是传入到 __call__ 中,同时 __init__ 用来初始化类装饰器的参数。

接下来我们使用类装饰器来复现第五章节中的效果:

class Func_5:    def __init__(self, name=None):        self.name = name    def __call__(self, func):        def wrapper():            func()            print('功能5由{}实现'.format(self.name))        return wrapper@Func_5('若水')def module():    print('功能1')    print('功能2')    print('功能3')    print('功能4')module()# 功能1# 功能2# 功能3# 功能4# 功能5由若水实现

七、内置装饰器

Python中有许多内置装饰器,这里仅介绍最常见的三种:@claSSMethod@staticmethod@property

7.1 @classmethod

@classmethod 用于装饰类中的函数,使用它装饰的函数不需要进行实例化也可调用。需要注意的是,被装饰的函数不需要 self 参数,但第一个参数需要是表示自身类的 cls 参数,它可以来调用类的属性,类的方法,实例化对象等。

cls 代表类本身,self 代表实例本身。

具体请看下例:

class A:    num = 100    def func1(self):        print('功能1')    @classmethod    def func2(cls):        print('功能2')        print(cls.num)        cls().func1()A.func2()# 功能2# 100# 功能1

7.2 @staticmethod

@staticmethod 同样用来修饰类中的方法,使用它装饰的函数的参数没有任何限制(即无需传入 self 参数),并且可以不用实例化调用该方法。当然,实例化后调用该方法也是允许的。

具体如下:

class A:    @staticmethod    def add(a, b):        return a + bprint(A.add(2, 3))# 5print(A().add(2, 3))# 5

7.3 @property

使用 @property 装饰器,我们可以直接通过方法名来访问类方法,不需要在方法名后添加一对 () 小括号。

class A:    @property    def printer(self):        print('Hello World')a = A()a.printer# Hello World

除此之外,@property 还可以用来防止类的属性被修改。考虑如下场景

class A:    def __init__(self):        self.name = 'ABC'a = A()print(a.name)# ABCa.name = 1print(a.name)# 1

可以看出类中的属性 name 可以被随意修改。如果要防止修改,则可以这样做

class A:    def __init__(self):        self.name_ = 'ABC'    @property    def name(self):        return self.name_        a = A()print(a.name)# ABCa.name = 1print(a.name)# AttributeError: can't set attribute

到此,关于“Python中的装饰器知识点有哪些”的学习就结束了,希望能够解决大家的疑惑。理论与实践的搭配能更好的帮助大家学习,快去试试吧!若想继续学习更多相关知识,请继续关注编程网网站,小编会继续努力为大家带来更多实用的文章!

--结束END--

本文标题: Python中的装饰器知识点有哪些

本文链接: https://lsjlt.com/news/341303.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • Python中的装饰器知识点有哪些
    这篇文章主要介绍“Python中的装饰器知识点有哪些”,在日常操作中,相信很多人在Python中的装饰器知识点有哪些问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Python中的装饰器知识点有哪些”的疑惑有所...
    99+
    2023-07-02
  • Python装饰器的相关知识点有哪些
    这篇文章主要介绍“Python装饰器的相关知识点有哪些”,在日常操作中,相信很多人在Python装饰器的相关知识点有哪些问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Python装饰器的相关知识点有哪些”的疑...
    99+
    2023-06-25
  • python装饰器相关知识点
    小编给大家分享一下python装饰器相关知识点,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!一、装饰器1.相关知识点*args:负责将多余的位置实参汇总,赋值给a...
    99+
    2023-06-29
  • Python装饰器的11个知识点分享
    本篇内容介绍了“Python装饰器的11个知识点分享”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!01. Hello,装饰器装饰器的使用方法...
    99+
    2023-06-02
  • Python中的pandas知识点有哪些
    本篇内容主要讲解“Python中的pandas知识点有哪些”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“Python中的pandas知识点有哪些”吧!前言pandas 是基于 Numpy 的一种...
    99+
    2023-06-27
  • Python中的PEP知识点有哪些
    本篇内容介绍了“Python中的PEP知识点有哪些”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!PEP是什么?PEP的全称是Python E...
    99+
    2023-06-02
  • Python小知识 - Python装饰器
    Python装饰器 在Python中,装饰器是一个特殊的函数,可以将其他函数包装在装饰器函数中,并且将被包装的函数作为参数传递给装饰器函数。 使用装饰器的好处是可以自动在被包装的函数前后执行一...
    99+
    2023-09-13
    Python YYDS
  • Python Pandas的知识点有哪些
    本篇内容介绍了“Python Pandas的知识点有哪些”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!为什么要学习Pandas那么问题来了:...
    99+
    2023-06-30
  • python中函数知识点有哪些
    这篇文章主要介绍python中函数知识点有哪些,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!函数参数的两大分类形式参数函数定义阶段括号所写的参数实际参数函数调用阶段括号内传入的参数形参与实参的关系可以将形参看成是变量...
    99+
    2023-06-25
  • python框架中flask的知识点有哪些
    小编给大家分享一下python框架中flask的知识点有哪些,希望大家阅读完这篇文章之后都有所收获,下面让我们一起去探讨吧!一、Flask蓝图目录我们之前写的Flask项目都是自己组织的目录结构,其实Flask官方有其推荐的目录结构,以下就...
    99+
    2023-06-15
  • Python编码的知识点有哪些
    这篇文章主要介绍“Python编码的知识点有哪些”,在日常操作中,相信很多人在Python编码的知识点有哪些问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Python编码的知识点有哪些”的疑惑有所帮助!接下来...
    99+
    2023-06-16
  • Python的基础知识点有哪些
    本篇内容介绍了“Python的基础知识点有哪些”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!python简介python是一种面向对象的解释...
    99+
    2023-06-02
  • Python的Cookie知识点有哪些呢
    这期内容当中小编将会给大家带来有关Python的Cookie知识点有哪些呢,文章内容丰富且以专业的角度为大家分析和叙述,阅读完这篇文章希望大家可以有所收获。相信很多同学肯定听过Cookie这个东西,也大概了解其作用,但是其原理以及如何设置,...
    99+
    2023-06-02
  • Python列表的知识点有哪些
    本篇内容主要讲解“Python列表的知识点有哪些”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“Python列表的知识点有哪些”吧!一、列表的创建方式# -*- coding:...
    99+
    2023-06-02
  • Python基本知识点有哪些
    本篇内容主要讲解“Python基本知识点有哪些”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“Python基本知识点有哪些”吧!Python注释python中单行注释采用 # 开头。python ...
    99+
    2023-06-29
  • Python基础知识点有哪些
    本篇内容主要讲解“Python基础知识点有哪些”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“Python基础知识点有哪些”吧!Python程序文件结构程序→模块→语句→表达式Python的关键要...
    99+
    2023-06-02
  • python端口知识点有哪些
    本篇内容介绍了“python端口知识点有哪些”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!1. 什么是端口端口就好一个房子的门,是出入这间房...
    99+
    2023-06-02
  • java中的HTML知识点有哪些
    这篇文章主要介绍“java中的HTML知识点有哪些”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“java中的HTML知识点有哪些”文章能帮助大家解决问题。   1...
    99+
    2024-04-02
  • Go中Sync.Map的知识点有哪些
    这篇文章主要讲解了“ Go中Sync.Map的知识点有哪些”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“ Go中Sync.Map的知识点有哪些”吧!sync.Map 优势在 Go 官方文档中...
    99+
    2023-06-15
  • Android中LayoutInflater的知识点有哪些
    这篇文章主要介绍“Android中LayoutInflater的知识点有哪些”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“Android中LayoutInflater的知识点有哪些”文章能帮助大家解...
    99+
    2023-06-29
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作