我的编程空间,编程开发者的网络收藏夹
学习永远不晚

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

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

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装饰器功能内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

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

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

python中装饰器的原理

装饰器这玩意挺有用,当时感觉各种绕,现在终于绕明白了,俺滴个大爷,还是要慢慢思考才能买明白各种的真谛,没事就来绕一绕 def outer(func): def inner(): print("认证成功")
2023-01-30

详解Python中最难理解的点-装饰器

本文将带领大家由浅入深的去窥探一下,这个装饰器到底是何方神圣,看完本篇,装饰器就再也不是难点了. 一、什么是装饰器 网上有人是这么评价装饰器的,我觉得写的很有趣,比喻的很形象每个人都有的内裤主要是用来遮羞,但是到了冬天它没法为我们防风御寒,
2022-06-04

深入理解Python中装饰器的用法

因为函数或类都是对象,它们也能被四处传递。它们又是可变对象,可以被更改。在函数或类对象创建后但绑定到名字前更改之的行为为装饰(decorator)。 “装饰器”后隐藏了两种意思——一是函数起了装饰作用,例如,执行真正的工作,另一个是依附于装
2022-06-04

python之我对装饰器的理解

从一开始学习python的时候,就一直不是很理解装饰器是个什么东东,再看了很多篇博文和自己动手敲了好多代码后,算是略有了解。  我理解的装饰器是: 在不改变原有函数调用的情况下,对其进行包装,使其变成另外一种函数来使用,一般的用途是 插入日
2023-01-31

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

python中的闭包从表现形式上定义(解释)为:如果在一个内部函数里,对在外部作用域(但不是在全局作用域)的变量进行引用,那么内部函数就被认为是闭包(closure)。 以下说明主要针对 python2.7,其他版本可能存在差异。 也许直接
2022-06-04

Python中如何理解和使用装饰器 @decorator

Python中如何理解和使用装饰器 @decorator,很多新手对此不是很清楚,为了帮助大家解决这个难题,下面小编将为大家详细讲解,有这方面需求的人可以来学习下,希望你能有所收获。Python的装饰器(decorator)是一个很棒的机制
2023-06-02

对Python装饰器的个人理解方法

0.说明                 在自己好好总结并对Python装饰器的执行过程进行分解之前,对于装饰器虽然理解它的基本工作方式,但对于存在复杂参数的装饰器(装饰器和函数本身都有参数),总是会感到很模糊,即使这会弄懂了,下一次也很快
2023-01-31

详解Python网络爬虫功能的基本写法

网络爬虫,即Web Spider,是一个很形象的名字。把互联网比喻成一个蜘蛛网,那么Spider就是在网上爬来爬去的蜘蛛。 1. 网络爬虫的定义 网络蜘蛛是通过网页的链接地址来寻找网页的。从网站某一个页面(通常是首页)开始,读取网页的内容,
2022-06-04

Python 如何理解又晕又好用的装饰器

Python 装饰器这东西对初学者来说是个坑,很容易绕晕,笔者当时初学装饰器时花费了数天时间,看了不同讲师对这块内容的讲解,还是一知半解。   不过装饰器在开发中可是很好用的,有必要攻破,希望这篇文章能帮助学习者快速攻破难关。初步理解# 先
2023-01-31

如何理解两个很实用的Python装饰器

本篇文章给大家分享的是有关如何理解两个很实用的Python装饰器,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。1.超时函数这个函数的作用在于可以给任意可能会hang住的函数添加
2023-06-21

Python编程中装饰器的使用示例解析

装饰函数和方法 我们先定义两个简单的数学函数,一个用来计算平方和,一个用来计算平方差:# get square sum def square_sum(a, b):return a**2 + b**2# get square diff def
2022-06-04

Python中装饰器的常见问题及解决方案

Python中装饰器的常见问题及解决方案什么是装饰器?装饰器是Python中一种非常强大的功能,可以用来修改已有函数或类的行为,而无需修改其源代码。装饰器实际上是个函数或类,它接受一个函数或类作为参数,然后返回一个新的函数或类。如何编写一个
2023-10-22

实例讲解Python编程中@property装饰器的用法

取值和赋值class Actress():def __init__(self):self.name = 'TianXin'self.age = 5类Actress中有两个成员变量name和age。在外部对类的成员变量的操作,主要包括取值和赋
2022-06-04

Python中的装饰器和上下文管理器是如何工作的?

Python中的装饰器和上下文管理器是如何工作的?在Python中,装饰器和上下文管理器是两个非常有用的概念和功能。它们都是为了简化代码、增加代码可读性以及方便代码的重用。一、装饰器装饰器是Python中一种用于修改函数的行为的特殊函数。它
2023-10-22

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录