python重难点之装饰器详解

2024-02-09 10:18
文章标签 python 详解 重难点 装饰

本文主要是介绍python重难点之装饰器详解,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

背景

虽然之前看过装饰器的相关内容,但是今天想起来,一直没有好好总结一下,所以特地记录下关于装饰器的一系列用法。
要想理解装饰器首先要明确颇python中的三个概念:
1.一切函数皆为对象
2.高阶函数
3.嵌套函数
然后才能理解:
4.什么是装饰器?
5.装饰器如何实现?
6.装饰器有什么用?

详细解释

一切函数皆为对象

准确来说在Python,一切皆为对象,此处说的点与函数相关所以将范围缩小了一下。看一个最基本的例子,我们可以发现我们可以直接将test1像变量一样赋值给a,然后a可以像函数一样使用。

def test1():print('i am test1')# 可将test1想变量一样赋值给a,然后a便可以像函数一样使用
a = test1
a()
# output:
# i am test1

高阶函数

函数参数可以接受变量,那么一个函数可以接受另一个函数作为参数,或者返回值为函数(这个稍后再说),那这种函数称之为高阶函数,如下面这段代码:

def add (a, b, f):return f(a) + f(b)
res = add(3, -6, abs)
print(res)

在这段代码代码中,add()就是一个高阶函数,可以看见在add()中的参数中有将abs()这个取绝对值函数做为参数。

嵌套函数

在一个函数的函数体内用def去申明一个函数,这样的函数叫做嵌套函数,如下面这段代码:

def foo():print('in the foo')def bar():print('in the bar')bar()
foo()
# outputs:
# in the foo
# in the bar

通过调用foo(),将在foo()内部定义并且调用bar()。我们是稍微改动一下代码:

def foo():print('in the foo')def bar():print('in the bar')return bar # 将bar作为foo()的返回值
a = foo()

上面这段代码稍微改了一下foo的返回值,即将bar作为foo的返回值返回了,还记得上面所说的高阶函数的定义么?这就是将函数返回的类型,然后我们将foo()赋值给变量a,这不就是我们呢所说的函数即变量的概念么?
最后得到输出:

in the foo

如果我们在后面再加一句:

a()

将会输出:

in the bar

这表明现在这个a已经是个函数类型了,他的功能就是foo()函数里面bar()的功能。

什么是装饰器?

说了这么多铺垫,那到底什么是装饰器呢?
其实装饰器的本质还是函数,它是为了装饰其他函数的,说白了就是为其他函数添加附加功能的
具体什么意思呢?比如我们我们之前写了一个函数,我们现在想在这个函数上添加一些功能,但是我们又不能改变在原来的函数基本上修改,而且还不能修改它的调用方式,因为它可能在很多地方已经被调用了,所以我们就必须要搞一个装饰器,来给原来的函数装饰一下,便于实现新的功能。
那装饰器应该怎么弄?我用一个公式来概括一下:
高阶函数 + 嵌套函数 ----> 装饰器
即通过高阶函数和嵌套函数我们就可以实现一个装饰器

如何实现一个装饰器

假设我们有一个函数func()如下,我现在想知道这个函数总共运行了多长时间,并且打印出来,而且我不能修改func()本身,并且不可以该变它的调用方式,那怎么办?

import timedef func():time.sleep(2) #模拟一系列的操作print('i am func in 1')

我们经过思考,结合上文可以写下如下的函数:

def func_time(func):start_time = time.time()func()end_time = time.time()print('func time is ',end_time - start_time)func_time(func) 
# outputs:
# i am func in 1
# func time is  2.0014798641204834

我们可以发现func_time()是可以统计func()的运行时间的,但是我想要的结果是直接使用func()就可以出现这样的效果,而且以后每次这么用,都会有这样的效果啊。
于是我们想起一切函数皆为对象,以及嵌套函数、高阶函数的用法,再改一改,得到如下的代码:

def func_time(func):def wrapper():start_time = time.time()func()end_time = time.time()print('func time is ',end_time - start_time)return wrapperfunc = func_time(func)
func()

我们在func_time()中使用嵌套函数定义了一个wrapper()函数,然后将刚才的操作都放在这个wrapper()中了,最后我们将wrapper作为func_time()的返回值返回了,在函数外面我们将func_time(func)又赋值给了func(),即将wrapper赋值给了func(),也就是说func()其实是实现的wrapper()的功能,而在wrapper中不但有func()的功能而且还有计算func()运行时间的功能。最后我们每次调用func()就会实现计时的功能了。
到这里其实我们已经手动的实现了python的装饰器功能了,但是有没有更简单的方法呢?是有的,在python中提供了一个装饰器语法,在上面的这个例子中,我们只需在func()函数前面加上一句@func_time。
@ 符号就是装饰器的语法糖,它放在函数开始定义的地方,这样就可以省略最后一步再次赋值的操作。什么是语法糖?就是计算机添加的某种语法,对语言的功能没有影响,但方便程序员使用
然后我们直接就可以使用func()即可,完整的也就是:

@func_time
def func():time.sleep(2)print('i am func in 1')

也就是说@func_time其实就是等价于

func = func_time(func)

还有一点需要注意的是func_time()这个函数的定义一定要写在func()定义前面,要不然使用@func_time,python在内存中是找不到func_time()的位置的。
完整的代码是这样的:

import timedef func_time(func):def wrapper():start_time = time.time()func()end_time = time.time()print('func time is ',end_time - start_time)return wrapper@func_time
def func():time.sleep(2)print('i am func in 1')func()

到此你就实现了一个基本的装饰器,但是如果你还不满足,请继续向下看。

-------------------------------------华丽的分割线------------------------------------------

如何实现一个装饰器(进阶)

装饰器还可以怎么用?首先第一个就是装饰器可以累计使用
现在我有一个函数:

def say():return "Hello"

我希望它可以根据不同的需要实现以下两种输出,不定时切换:

<b><i>Hello</i></b>
<i><b>Hello</b></i>

我们可以用装饰器很轻易的实现,我们先实现两个装饰器:

# 用来装饰say()产生<b></b>
def makebold(fn):def wrapper():return "<b>" + fn() + "</b>"return wrapper# 用来装饰say()产生<i></i>
def makeitalic(fn):def wrapper():return "<i>" + fn() + "</i>"return wrapper

然后我们来装饰say():

@makebold
@makeitalic
def say():return "hello"print(say())

通过上面的装饰我们可以输出:

<b><i>hello</i></b>

如果我们将两个装饰器的位置改变一下即:

@makeitalic
@makebold
def say():return "hello"print(say())

我们就可以输出:

<i><b>hello</b></i>

如何实现一个装饰器(高阶)

我们应该如何给一个被修饰的函数传递参数呢?
其实当你调用被装饰器返回的函数时,实际你是在调用封装函数 ,向封装函数传递参数可以让封装函数把参数传递给被装饰的函数。
我们先来看看在最开始统计函数运行时间的小程序上做的一点改变后的样子:

def count(func):def sleep_time(name): # 传入参数start_time = time.time()func(name)stop_time = time.time()print(stop_time - start_time)return sleep_time@count
def sleep2(name):time.sleep(2)print(name, 'sleepping in 2') #打印参数sleep2('sty')
# outputs:
# sty sleepping in 2
# 2.002162218093872

我们可以看到’sty’已经作为参数传进入了,但是如果我们有的时候我们对有的被装饰函数我们不需要参数,或者我们不确定有多少个参数该怎么办?我们总不能再重复的写装饰器吧?这个时候我们就需要使用到*args,**kwargs了,在python中参数有四类:必选参数、默认参数、可变参数和关键字参数,这4种参数都可以一起使用,或者只用其中某些,但是请注意,参数定义的顺序必须是:必选参数、默认参数、可变参数和关键字参数,关于具体的详细介绍点这里。

*args:接受N个位置参数,转换为元组的形式
**kwargs:接受N个位置参数,转换为字典的形式

这样我们就可以对我们的装饰器进行一定的改进了,如下面的代码:

import timedef count(func):def sleep_time(*args,**kwargs):start_time = time.time()func(*args,*kwargs)stop_time = time.time()print(stop_time - start_time)return sleep_time
@count
def sleep1():time.sleep(2)print('i am sleepping in 1')@count
def sleep2(name):time.sleep(2)print(name, 'sleepping in 2')sleep1()
sleep2('sty')
# outputs:
# i am sleepping in 1
# 2.0013535022735596
# sty sleepping in 2
# 2.0010647773742676

在这段代码中我们利用装饰器装饰了两个函数sleep()和sleep1(),sleep()没有传参数,sleep1()传了一个参数,发现都可以完好运行。

一个问题需要引起你的注意:
在上面的代码最后添加下面的两行,我们打印下sleep1()和sleep2()的名字

print('sleep1 name is :', sleep1.__name__)
print('sleep2 name is :', sleep2.__name__)

得到的结果是:

sleep1 name is : sleep_time
sleep2 name is : sleep_time

虽然我们知道sleep1()和sleep2()就是执行的sleep_time()的功能,但是我们还是不愿意看见它的真实名字改变。这并不是我们想要的!我们希望的结果输出应该是:

sleep1 name is : sleep1
sleep2 name is : sleep2

这里的函数被sleep_time()替代了。它重写了我们函数的名字和注释文档(docstring)。幸运的是Python提供给我们一个简单的函数来解决这个问题,那就是functools.wraps。我们修改一下上例, 在封装函数前加上@wraps(func),可以得到:

from functools import wraps
import timedef count(func):@wraps(func)    #在封装函数前加上def sleep_time(*args,**kwargs):start_time = time.time()func(*args,*kwargs)stop_time = time.time()print(stop_time - start_time)return sleep_time
@count
def sleep1():time.sleep(2)print('i am sleepping in 1')@count
def sleep2(name):time.sleep(2)print(name, 'sleepping in 2')sleep1()
sleep2('sty')
print('sleep1 name is :', sleep1.__name__)
print('sleep2 name is :', sleep2.__name__)
#Outputs:
# i am sleepping in 1
# 2.0020651817321777
# sty sleepping in 2
# 2.002112627029419
# sleep1 name is : sleep1
# sleep2 name is : sleep2

装饰器实战之授权认证

需求:假如现在一个网站有三个页面,index, home, bbs, 其中index页面是你不需要登录就可以查看的,而home,bbs是需要登录才能查看的,并且告诉我登录授权形式,有local和remote两种可以选择,我们应该怎么做?

解决方法:装饰器高阶用法,需要给装饰器一开始就传参数,需要在装饰器中再写一个函数传递被装饰函数。说的可能有点绕,具体查看下面的代码:

from functools import wrapsuser, passwd = 'sty', '1234'  # 定义一个默认的用户名和密码
def auth(auth_type):print('now auth_type is:', auth_type)def outer_wrapper(func): # 又设了一层函数,来传递被装饰函数的@wraps(func)def wrapper(*args, **kwargs):user_name = input("UserName:").strip()pass_word = input("PassWord:").strip()if user == user_name and passwd == pass_word:print("\033[32;1mUser has passed authentication\033[0m")  #让打印带颜色显示func(*args, **kwargs)else:print("\033[31;1mInvalid username or passward\033[0m")return wrapper  return outer_wrapperdef index():print("welcome to the index page")@auth(auth_type='local') #等价于home = auth(auth_type='local')(home)
def home():print("welcome to the home page")@auth(auth_type='remote') #等价于home = auth(auth_type='ldap')(home)
def bbs():print("welcome to the bbs page")index()
print('login in home, input name and password:')
home()
print('login in bbs: input name and password:')
bbs()

我们注意到:
@auth(auth_type=‘local’) 等价于home = auth(auth_type=‘local’)(home)
@auth(auth_type=‘remote’) 等价于home = auth(auth_type=‘ldap’)(home)
当这样看的时候我们就不难理解它是如何实现的了
到此装饰器的几乎大部分功能就差不多了,其他的一些具体用法还需要你逐渐去探索

最后提一句

在学习装饰器的时候,如果你还想从更加深入的去理解,那你就得需要知道’闭包’,关于什么是闭包?以及它和装饰器的关系?有时间将在接下来的博客中提到。

参考文档:

https://stackoverflow.com/questions/739654/how-to-make-a-chain-of-function-decorators
https://stackoverflow.com/questions/13857/can-you-explain-closures-as-they-relate-to-python
https://segmentfault.com/a/1190000004461404
http://www.cnblogs.com/itech/archive/2011/12/31/2308640.html
http://www.cnblogs.com/vamei/archive/2012/12/15/2772451.html

转载请注明出处:
CSDN:楼上小宇_home:http://blog.csdn.net/sty945
简书:楼上小宇:http://www.jianshu.com/u/1621b29625df

这篇关于python重难点之装饰器详解的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/693930

相关文章

基于Python编写一个git自动上传的脚本(打包成exe)

《基于Python编写一个git自动上传的脚本(打包成exe)》这篇文章主要为大家详细介绍了如何基于Python编写一个git自动上传的脚本并打包成exe,文中的示例代码讲解详细,感兴趣的小伙伴可以跟... 目录前言效果如下源码实现利用pyinstaller打包成exe利用ResourceHacker修改e

Python在二进制文件中进行数据搜索的实战指南

《Python在二进制文件中进行数据搜索的实战指南》在二进制文件中搜索特定数据是编程中常见的任务,尤其在日志分析、程序调试和二进制数据处理中尤为重要,下面我们就来看看如何使用Python实现这一功能吧... 目录简介1. 二进制文件搜索概述2. python二进制模式文件读取(rb)2.1 二进制模式与文本

Python中Tkinter GUI编程详细教程

《Python中TkinterGUI编程详细教程》Tkinter作为Python编程语言中构建GUI的一个重要组件,其教程对于任何希望将Python应用到实际编程中的开发者来说都是宝贵的资源,这篇文... 目录前言1. Tkinter 简介2. 第一个 Tkinter 程序3. 窗口和基础组件3.1 创建窗

基于C++的UDP网络通信系统设计与实现详解

《基于C++的UDP网络通信系统设计与实现详解》在网络编程领域,UDP作为一种无连接的传输层协议,以其高效、低延迟的特性在实时性要求高的应用场景中占据重要地位,下面我们就来看看如何从零开始构建一个完整... 目录前言一、UDP服务器UdpServer.hpp1.1 基本框架设计1.2 初始化函数Init详解

Django调用外部Python程序的完整项目实战

《Django调用外部Python程序的完整项目实战》Django是一个强大的PythonWeb框架,它的设计理念简洁优雅,:本文主要介绍Django调用外部Python程序的完整项目实战,文中通... 目录一、为什么 Django 需要调用外部 python 程序二、三种常见的调用方式方式 1:直接 im

Python字符串处理方法超全攻略

《Python字符串处理方法超全攻略》字符串可以看作多个字符的按照先后顺序组合,相当于就是序列结构,意味着可以对它进行遍历、切片,:本文主要介绍Python字符串处理方法的相关资料,文中通过代码介... 目录一、基础知识:字符串的“不可变”特性与创建方式二、常用操作:80%场景的“万能工具箱”三、格式化方法

springboot+redis实现订单过期(超时取消)功能的方法详解

《springboot+redis实现订单过期(超时取消)功能的方法详解》在SpringBoot中使用Redis实现订单过期(超时取消)功能,有多种成熟方案,本文为大家整理了几个详细方法,文中的示例代... 目录一、Redis键过期回调方案(推荐)1. 配置Redis监听器2. 监听键过期事件3. Redi

Springboot配置文件相关语法及读取方式详解

《Springboot配置文件相关语法及读取方式详解》本文主要介绍了SpringBoot中的两种配置文件形式,即.properties文件和.yml/.yaml文件,详细讲解了这两种文件的语法和读取方... 目录配置文件的形式语法1、key-value形式2、数组形式读取方式1、通过@value注解2、通过

浅析python如何去掉字符串中最后一个字符

《浅析python如何去掉字符串中最后一个字符》在Python中,字符串是不可变对象,因此无法直接修改原字符串,但可以通过生成新字符串的方式去掉最后一个字符,本文整理了三种高效方法,希望对大家有所帮助... 目录方法1:切片操作(最推荐)方法2:长度计算索引方法3:拼接剩余字符(不推荐,仅作演示)关键注意事

自定义注解SpringBoot防重复提交AOP方法详解

《自定义注解SpringBoot防重复提交AOP方法详解》该文章描述了一个防止重复提交的流程,通过HttpServletRequest对象获取请求信息,生成唯一标识,使用Redis分布式锁判断请求是否... 目录防重复提交流程引入依赖properties配置自定义注解切面Redis工具类controller