Python高级编程之闭包和装饰器
创始人
2025-05-31 02:09:14
0

闭包和装饰器

第一部分 闭包

1. 闭包的介绍

我们知道当函数调用完,函数内定义的变量都销毁了,但是我们有时候需要保存函数内的这个变量,每次在这个变量的基础上完成一些列的操作,比如: 每次在这个变量的基础上和其它数字进行求和计算,那怎么办呢?

我们就可以通过闭包来解决这个需求。

闭包的定义:

在函数嵌套的前提下,内部函数使用了外部函数的变量,并且外部函数返回了内部函数,我们把这个使用外部函数变量的内部函数称为闭包。

2. 闭包的构成条件

通过闭包的定义,我们可以得知闭包的形成条件:

在函数嵌套(函数里面再定义函数)的前提下
内部函数使用了外部函数的变量(还包括外部函数的参数)
外部函数返回了内部函数

3. 简单闭包的示例代码

# 定义一个外部函数
def func_out(num1):# 定义一个内部函数def func_inner(num2):# 内部函数使用了外部函数的变量(num1)result = num1 + num2print("结果是:", result)# 外部函数返回了内部函数,这里返回的内部函数就是闭包return func_inner# 创建闭包实例    
f = func_out(1)
# 执行闭包
f(2)
f(3)

执行结果:

结果是: 3
结果是: 4

闭包执行结果的说明:

通过上面的输出结果可以看出闭包保存了外部函数内的变量num1,每次执行闭包都是在num1 = 1 基础上进行计算。

4. 闭包的作用

闭包可以保存外部函数内的变量,不会随着外部函数调用完而销毁。
注意点:

由于闭包引用了外部函数的变量,则外部函数的变量没有及时释放,消耗内存。

5. 小结

1.当返回的内部函数使用了外部函数的变量就形成了闭包
2.闭包可以对外部函数的变量进行保存
3.实现闭包的标准格式:

 # 外部函数def test1(a):b = 10# 内部函数def test2():# 内部函数使用了外部函数的变量或者参数print(a, b)# 返回内部函数, 这里返回的内部函数就是闭包实例return test2

第二部分 闭包的使用

1. 案例

需求: 根据配置信息使用闭包实现不同人的对话信息,例如对话:

张三: 到北京了吗? 李四: 已经到了,放心吧。

2. 实现步骤说明

定义外部函数接收不同的配置信息参数,参数是人名
定义内部函数接收对话信息参数
在内部函数里面把配置信息和对话信息进行拼接输出

3. 功能代码的实现

# 外部函数
def config_name(name):# 内部函数def say_info(info):print(name + ": " + info)return say_infotom = config_name("Tom")tom("你好!")
tom("你好, 在吗?")jerry = config_name("jerry")jerry("不在, 不和玩!")

运行结果:

Tom: 你好!
Tom: 你好, 在吗?
jerry: 不在, 不和玩!

闭包案例说明:

闭包还可以提高代码的可重用性,不需要再手动定义额外的功能函数。

5. 小结

闭包不仅可以保存外部函数的变量还可以提高代码的可重用行。

第三部分 修改闭包内使用的外部变量

1. 修改闭包内使用的外部变量

修改闭包内使用的外部变量的错误示例:

# 定义一个外部函数
def func_out(num1):# 定义一个内部函数def func_inner(num2):# 这里本意想要修改外部num1的值,实际上是在内部函数定义了一个局部变量num1num1 = 10# 内部函数使用了外部函数的变量(num1)result = num1 + num2print("结果是:", result)print(num1)func_inner(1)print(num1)# 外部函数返回了内部函数,这里返回的内部函数就是闭包return func_inner# 创建闭包实例
f = func_out(1)
# 执行闭包
f(2)

修改闭包内使用的外部变量的错误示例:

# 定义一个外部函数
def func_out(num1):# 定义一个内部函数def func_inner(num2):# 这里本意想要修改外部num1的值,实际上是在内部函数定义了一个局部变量num1nonlocal num1  # 告诉解释器,此处使用的是 外部变量a# 修改外部变量num1num1 = 10# 内部函数使用了外部函数的变量(num1)result = num1 + num2print("结果是:", result)print(num1)func_inner(1)print(num1)# 外部函数返回了内部函数,这里返回的内部函数就是闭包return func_inner# 创建闭包实例
f = func_out(1)
# 执行闭包
f(2)

2. 小结

修改闭包内使用的外部函数变量使用 nonlocal 关键字来完成。

第四部分 装饰器

1. 装饰器的定义

就是给已有函数增加额外功能的函数,它本质上就是一个闭包函数。

装饰器的功能特点:

不修改已有函数的源代码
不修改已有函数的调用方式
给已有函数增加额外的功能

2. 装饰器的示例代码

# 添加一个登录验证的功能
def check(fn):def inner():print("请先登录....")fn()return innerdef comment():print("发表评论")# 使用装饰器来装饰函数
comment = check(comment)
comment()# 装饰器的基本雏形
# def decorator(fn): # fn:目标函数.
#     def inner():
#         '''执行函数之前'''
#         fn() # 执行被装饰的函数
#         '''执行函数之后'''
#     return inner

代码说明:

闭包函数有且只有一个参数,必须是函数类型,这样定义的函数才是装饰器。
写代码要遵循开放封闭原则,它规定已经实现的功能代码不允许被修改,但可以被扩展。
执行结果:

请先登录....
发表评论

3. 装饰器的语法糖写法

如果有多个函数都需要添加登录验证的功能,每次都需要编写func = check(func)这样代码对已有函数进行装饰,这种做法还是比较麻烦。

Python给提供了一个装饰函数更加简单的写法,那就是语法糖,语法糖的书写格式是: @装饰器名字,通过语法糖的方式也可以完成对已有函数的装饰

# 添加一个登录验证的功能
def check(fn):print("装饰器函数执行了")def inner():print("请先登录....")fn()return inner# 使用语法糖方式来装饰函数
@check
def comment():print("发表评论")comment()

说明:

@check 等价于 comment = check(comment)
装饰器的执行时间是加载模块时立即执行。
执行结果:

请先登录....
发表评论
  1. 小结
    装饰器本质上就是一个闭包函数,它可以对已有函数进行额外的功能扩展。
    装饰器的语法格式:
# 装饰器
# def decorator(fn): # fn:被装饰的目标函数.
#     def inner():
#         '''执行函数之前'''
#         fn() # 执行被装饰的目标函数
#         '''执行函数之后'''
#     return inner

装饰器的语法糖用法: @装饰器名称,同样可以完成对已有函数的装饰操作。

第五部分 装饰器的使用

1. 装饰器的使用场景

函数执行时间的统计
输出日志信息

2. 装饰器实现已有函数执行时间的统计

import time# 装饰器函数
def get_time(func):def inner():begin = time.time()func()end = time.time()print("函数执行花费%f" % (end-begin))return inner@get_time
def func1():for i in range(100000):print(i)func1()

执行结果:

...
99995
99996
99997
99998
99999
函数执行花费0.329066
  1. 小结
    通过上面的示例代码可以得知装饰器的作用:

在不改变已有函数源代码及调用方式的前提下,对已有函数进行功能的扩展。

第六部分 通用装饰器的使用

1. 装饰带有参数的函数

# 添加输出日志的功能
def logging(fn):def inner(num1, num2):print("--正在努力计算--")fn(num1, num2)return inner# 使用装饰器装饰函数
@logging
def sum_num(a, b):result = a + bprint(result)sum_num(1, 2)

执行结果

--正在努力计算--
3

2. 装饰带有返回值的函数

# 添加输出日志的功能
def logging(fn):def inner(num1, num2):print("--正在努力计算--")result = fn(num1, num2)return resultreturn inner# 使用装饰器装饰函数
@logging
def sum_num(a, b):result = a + breturn resultresult = sum_num(1, 2)
print(result)

执行结果

--正在努力计算--
3

3. 装饰带有不定长参数的函数

# 添加输出日志的功能
def logging(fn):def inner(*args, **kwargs):print("--正在努力计算--")fn(*args, **kwargs)return inner# 使用语法糖装饰函数
@logging
def sum_num(*args, **kwargs):result = 0for value in args:result += valuefor value in kwargs.values():result += valueprint(result)sum_num(1, 2, a=10)

执行结果

--正在努力计算--
13

4. 通用装饰器

# 添加输出日志的功能
def logging(fn):def inner(*args, **kwargs):print("--正在努力计算--")result = fn(*args, **kwargs)return resultreturn inner# 使用语法糖装饰函数
@logging
def sum_num(*args, **kwargs):result = 0for value in args:result += valuefor value in kwargs.values():result += valuereturn result@logging
def subtraction(a, b):result = a - bprint(result)result = sum_num(1, 2, a=10)
print(result)subtraction(4, 2)

执行结果

--正在努力计算--
13
--正在努力计算--
2

5. 小结

通用装饰器的语法格式:

# 通用装饰器
def logging(fn):def inner(*args, **kwargs):print("--正在努力计算--")result = fn(*args, **kwargs)return resultreturn inner

第七部分 多个装饰器的使用

1. 多个装饰器的使用示例代码

def make_div(func):"""对被装饰的函数的返回值 div标签"""def inner(*args, **kwargs):return "
" + func() + "
"return innerdef make_p(func):"""对被装饰的函数的返回值 p标签"""def inner(*args, **kwargs):return "

" + func() + "

"return inner# 装饰过程: 1 content = make_p(content) 2 content = make_div(content) # content = make_div(make_p(content)) @make_div @make_p def content():return "人生苦短"result = content()print(result)

代码说明:

多个装饰器的装饰过程是: 离函数最近的装饰器先装饰,然后外面的装饰器再进行装饰,由内到外的装饰过程

2. 小结

多个装饰器可以对函数进行多个功能的装饰,装饰顺序是由内到外的进行装饰

第八部分 带有参数的装饰器

1. 带有参数的装饰器介绍

带有参数的装饰器就是使用装饰器装饰函数的时候可以传入指定参数,语法格式: @装饰器(参数,…)

错误写法:

def decorator(fn, flag):def inner(num1, num2):if flag == "+":print("--正在努力加法计算--")elif flag == "-":print("--正在努力减法计算--")result = fn(num1, num2)return resultreturn inner@decorator('+')
def add(a, b):result = a + breturn resultresult = add(1, 3)
print(result)

执行结果:

Traceback (most recent call last):File "/home/python/Desktop/test/hho.py", line 12, in @decorator('+')
TypeError: decorator() missing 1 required positional argument: 'flag'

代码说明:

装饰器只能接收一个参数,并且还是函数类型。
正确写法:

在装饰器外面再包裹上一个函数,让最外面的函数接收参数,返回的是装饰器,因为@符号后面必须是装饰器实例。

# 添加输出日志的功能
def logging(flag):def decorator(fn):def inner(num1, num2):if flag == "+":print("--正在努力加法计算--")elif flag == "-":print("--正在努力减法计算--")result = fn(num1, num2)return resultreturn inner# 返回装饰器return decorator# 使用装饰器装饰函数
@logging("+")
def add(a, b):result = a + breturn result@logging("-")
def sub(a, b):result = a - breturn resultresult = add(1, 2)
print(result)result = sub(1, 2)
print(result)
  1. 小结
    使用带有参数的装饰器,其实是在装饰器外面又包裹了一个函数,使用该函数接收参数,返回是装饰器,因为 @ 符号需要配合装饰器实例使用

第九部分 类装饰器的使用

1. 类装饰器的介绍

装饰器还有一种特殊的用法就是类装饰器,就是通过定义一个类来装饰函数。

类装饰器示例代码:

class Check(object):def __init__(self, fn):# 初始化操作在此完成self.__fn = fn# 实现__call__方法,表示对象是一个可调用对象,可以像调用函数一样进行调用。def __call__(self, *args, **kwargs):# 添加装饰功能print("请先登陆...")self.__fn()@Check
def comment():print("发表评论")comment()

说明:
@Check 等价于 comment = Check(comment), 所以需要提供一个init方法,并多增加一个fn参数。
要想类的实例对象能够像函数一样调用,需要在类里面使用call方法,把类的实例变成可调用对象(callable),也就是说可以像调用函数一样进行调用。
在call方法里进行对fn函数的装饰,可以添加额外的功能。
执行结果:

请先登陆...
发表评论

2. 小结

想要让类的实例对象能够像函数一样进行调用,需要在类里面使用call方法,把类的实例变成可调用对象(callable)
类装饰器装饰函数功能在call方法里面进行添加

相关内容

热门资讯

AWSECS:访问外部网络时出... 如果您在AWS ECS中部署了应用程序,并且该应用程序需要访问外部网络,但是无法正常访问,可能是因为...
AWSElasticBeans... 在Dockerfile中手动配置nginx反向代理。例如,在Dockerfile中添加以下代码:FR...
银河麒麟V10SP1高级服务器... 银河麒麟高级服务器操作系统简介: 银河麒麟高级服务器操作系统V10是针对企业级关键业务...
北信源内网安全管理卸载 北信源内网安全管理是一款网络安全管理软件,主要用于保护内网安全。在日常使用过程中,卸载该软件是一种常...
AWR报告解读 WORKLOAD REPOSITORY PDB report (PDB snapshots) AW...
AWS管理控制台菜单和权限 要在AWS管理控制台中创建菜单和权限,您可以使用AWS Identity and Access Ma...
​ToDesk 远程工具安装及... 目录 前言 ToDesk 优势 ToDesk 下载安装 ToDesk 功能展示 文件传输 设备链接 ...
群晖外网访问终极解决方法:IP... 写在前面的话 受够了群晖的quickconnet的小水管了,急需一个新的解决方法&#x...
不能访问光猫的的管理页面 光猫是现代家庭宽带网络的重要组成部分,它可以提供高速稳定的网络连接。但是,有时候我们会遇到不能访问光...
Azure构建流程(Power... 这可能是由于配置错误导致的问题。请检查构建流程任务中的“发布构建制品”步骤,确保正确配置了“Arti...