‘闭包‘, ‘装饰器‘及其应用场景

news2025/3/24 18:52:43

‘闭包’, '装饰器’及其应用场景

一, 闭包及其应用场景

图解

在这里插入图片描述

闭包的定义

概述: 内部函数 使用了 外部函数 的变量, 这种写法就称之为闭包.
格式:

def 外部函数名(形参列表):
	外部函数的(局部)变量
	def 内部函数名(形参列表):
		内部函数的(局部)变量

	return 内部函数名

前提条件:

  1. 有嵌套. 外部函数嵌套内部函数
  2. 有引用. 内部函数引用外部函数的变量
  3. 有返回. 外部函数中, 返回 内部函数名(对象)

细节:
函数名函数名() 是两个概念, 前者表示 函数对象 , 后者表示 调用函数, 获取返回值 .

案例

案例(1): 观察函数名函数名()
案例需求

创建函数, 分别打印函数名函数名()
证明: 函数名 -> 是对象, 函数名() -> 是调用函数, 获取返回值

实现思路
  1. 创建函数.
  2. 打印函数名.
  3. 打印函数名().
  4. 将函数名赋值给变量, 这个变量就是: 函数对象.
  5. 打印函数对象.
  6. 打印函数对象().
# 1. 创建函数.
def get_sum(a, b):
	return a + b
# 2. 打印函数名.
print(get_sum)	# <function get_sum at 0x000001B4AC3DD800>, 对象.
# 3. 打印函数名().
print(get_sum(10, 20))		# 调用函数, 获取返回值.

# 4. 将函数名赋值给变量, 这个变量就是: 函数对象.
my_sum = get_sum
# 5. 打印函数对象.
print(my_sum)	# <function get_sum at 0x00000251CE53D800>
print(my_sum(100, 200))	# 300
案例(2): 闭包写法
案例需求

定义求和的闭包, 外部函数有参数num1, 内部函数有参数num2, 调用, 求解两数之和, 观察结果.

实现思路
  1. 定义外部函数.
  2. 定义内部函数.
  3. 求和.
  4. 调用上述函数.
  5. 另一种调用方式.
# 1. 定义外部函数.
def fn_outer(num1):
	# 2. 定义内部函数.
	def fn_inner(num2):					# 有嵌套
		# 3. 求和
		sum = num1 + num2				# 有引用
		print(f'求和结果: {sum}')
	return fn_inner						# 有返回

# 4. 调用上述的函数.
fn_inner = fn_outer(10)
fn_inner(20)
# 5. 另一种调用方式.
fn_inner(100)(200)

二, nonlocal关键字介绍

nonlocal关键字的定义

nonlocal关键字是Python内置的关键字, 可以实现 在内部函数中 修改外部函数的 变量值.

图解

在这里插入图片描述

案例需求

编写1个闭包, 让内部函数访问外部函数的参数 a = 100, 并观察结果.

实现思路

  1. 定义外部函数.
  2. 定义外部函数的(局部)变量.
  3. 定义内部函数, 访问外部函数的变量.
  4. 在内部函数中修改外部函数的变量.
  5. nonlocal: 可以实现在内部函数修改外部函数的变量值.
  6. 打印外部函数的变量.
  7. 返回 内部函数名(对象).
# 1. 定义外部函数.
def fn_outer()
	# 2. 定义外部函数的(局部)变量.
	a = 100
	# 3. 定义内部函数, 访问外部函数的变量.
	def fn_inner()
		# 4. 在内部函数中修改外部函数的变量.
		# 5. nonlocal: 可以实现在内部函数中修改外部函数的变量值.
		nonlocal a
		a = a + 1
		# 6. 打印外部函数的变量.
		print(f'a: {a}')

	# 7. 返回 内部函数名(对象)
	return fn_inner

# 8. 调用函数
fn_inner = fn_outer()
fn_inner()	# 101
fn_inner()	# 102
fn_inner()	# 103
		

三, 装饰器及其应用场景

装饰器的定义

概述:

装饰器的本质是1个闭包函数, 目的是 在不改变原有函数的基础上, 对1其功能做增强.
大白话: 装修队 在不改变房屋结构的情况下, 对房屋做装饰(功能增强).

提前条件:
  1. 有嵌套.
  2. 有引用.
  3. 有返回.
  4. 有额外功能.
格式:
  • 方法一: 传统写法.
修饰后的函数名 = 装饰器名(被装饰的原函数名)
装饰后的函数名()
  • 方法二: 语法糖.
在要被装饰的原函数上, 直接写 @装饰器名, 之后直接调用原函数即可.
图解:在这里插入图片描述

案例

案例(1): 发表评论.
案例需求

在发表评论前, 都是需要先登录的.

实现思路
  1. 定义外部函数, 形参列表接收 要被装饰的函数名(对象)
  2. 定义内部函数.
  3. 额外功能.
  4. 访问原函数, 即: 外部函数的引用.
  5. 返回内部函数对象.
  6. 定义函数, 表示 发表评论.
  7. 定义函数, 表示 充值中…
  8. 传统方式.
  9. 语法糖方式.
# 1.定义外部函数, 形参列表接收 要被装饰的函数名(对象)
def check_login(fn_name):       # fn_name: 被装饰的函数名(对象)
    # 2. 定义内部函数.
    def fn_inner():             # 有嵌套
        # 3. 额外功能
        print('校验登陆... 登陆成功!')
        # 4. 访问原函数, 即: 外部函数的引用.
        fn_name()               # 有引用
    # 5.  返回内部函数对象.
    return fn_inner             # 有返回

# 6. 定义函数, 表示 发表评论.
def comment():
    print("发表评论")
# 7. 定义函数, 表示 充值中....
@check_login    # 底层其实是: payment = check_login(payment)
def payment():
    print('充值中...')

# 8. 传统方式.
comment = check_login(comment)
comment()
# 9. 语法糖方式.
# payment = check_login(payment)
# payment()
payment()

案例(2): 装饰器装饰_无参无返回的原函数
细节

装饰器的内部函数格式 要和 被装饰的原函数 保持一致.
即:原函数是无参无返回的, 则 装饰器的内部函数也必须是 无参无返回的.
原函数有参有返回的, 则 装饰器的内部函数也必须是 有参有返回的.

案例需求

装饰器装饰_无参无返回的原函数

实现思路
# 需求: 定义无参无返回值的 get_sum()求和函数, 在不改变其代码的基础上, 添加友好提示: 正在努力计算中...
# 1. 定义装饰器.
def my_decorator(fn_name):
    # 2. 定义内部函数, 其格式必须和 被装饰的原函数 保持一致.
    def fn_inner():                 # 有嵌套
        # 3. 添加提示信息(额外功能)
        print('正在努力计算中...')     # 有额外功能
        # 4. 调用原函数.
        fn_name()                   # 有引用
    # 5. 返回内部函数(对象)
    return fn_inner                 # 有返回


# 6. 定义原函数.
@my_decorator		# @装饰器名
def get_sum():
    a = 10
    b = 20
    sum = a + b
    print(f'sum求和结果: {sum}')


# 7. 传统方式.
# get_sum = my_decorator(get_sum)
# get_sum()

# 8. 语法糖方式.
get_sum()
案例(3): 装饰器装饰_有参无返回值的原函数
细节

装饰器的内部函数格式 要和 被装饰的原函数 保持一致.
即:原函数是无参无返回的, 则 装饰器的内部函数也必须是 无参无返回的.
原函数有参有返回的, 则 装饰器的内部函数也必须是 有参有返回的.

案例需求

装饰器装饰_有参无返回值的原函数

实现思路
# 需求: 定义有参无返回值的 get_sum()求和函数, 在不改变其代码的基础上, 添加友好提示: 正在努力计算中...
# 1. 定义装饰器.
def my_decorator(fn_name):
    # 2. 定义内部函数
    def fn_inner(x, y):
        # 3. 额外功能
        print('正在努力计算中...')
        # 4. 调用原函数.
        fn_name(x, y)
    # 5. 返回内部函数.
    return fn_inner

# 6. 定义原函数, 有参无返回值.
@my_decorator
def get_sum(a, b):
    sum = a + b
    print(f'sum求和结果: {sum}')


# 7. 传统方式.
# get_sum = my_decorator(get_sum)
# get_sum(10, 20)

# 8. 语法糖方式.
get_sum(10, 20)

四, 总结

  • Q1:闭包的定义
    • 概述: 内部函数 使用了 外部函数 的变量, 这种写法就称之为闭包.
    • 格式:

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2320116.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

IDEA 快捷键ctrl+shift+f 无法全局搜索内容的问题及解决办法

本篇文章主要讲解IDEA、phpStrom、webStrom、pyCharm等jetbrains系列编辑器无法进行全局搜索内容问题的主要原因及解决办法。 日期&#xff1a;2025年3月22日 作者&#xff1a;任聪聪 现象描述&#xff1a; 1.按下ctrlshiftf 输入法转为了繁体。 2.快捷键ctrlshiftr 可以全局检…

Powershell WSL导出导入ubuntu22.04.5子系统

导出Linux子系统 导出位置在C盘下,根据自己的实际情况更改即可Write-Host "export ubuntu22.04.5" -ForegroundColor Green wsl --export Ubuntu-22.04 c:\Ubuntu-22.04.tar 导入Linux子系统 好处是目录可用在任意磁盘路径,便于迁移不同的设备之间Write-Host &quo…

论文笔记(七十三)Gemini Robotics: Bringing AI into the Physical World

Gemini Robotics: Bringing AI into the Physical World 文章概括1. 引言2. Gemini 2.0的具身推理2.1. 具身推理问答&#xff08;ERQA&#xff09;基准测试2.2. Gemini 2.0的具身推理能力2.3. Gemini 2.0支持零样本和少样本机器人控制 3. 使用 Gemini Robotics 执行机器人动作3…

AI + 医疗 Qwq大模型离线本地应用

通义千问Qwq-32b-FP16可用于社区医院、乡镇卫生院、诊所等小型医疗机构&#xff0c;替代专业合理用药系统&#xff0c;作为药品知识库&#xff0c;实现以下功能&#xff1a; 药品信息智能查询&#xff1a;检索药品的详细说明书、适应症、禁忌症、不良反应及药物相互作用等关键信…

元音辅音及其字母组合发音

文章目录 单元音长元音/ɑː//ɔ://u://i://ɜː/// 短元音/ʌ//ɒ//ʊ//ɪ//ə//e/ 双元音/eɪ//aɪ//ɔɪ//ɪə//eə//ʊə//əʊ//aʊ/ 辅音3个鼻辅音m n ŋ 5个独立浊辅音w j r l h 20个清浊相对的辅音s zʃ ʒf vθ p bt dk gts dztʃ dʒtr dr 以下是列举的部分字母组合…

【Vitis AIE】FPGA图像处理 11 双线性插值 Bilinear Interpolation

双线性插值 https://github.com/Xilinx/Vitis-Tutorials/tree/2024.2/AI_Engine_Development/AIE/Design_Tutorials/11-Bilinear_Interpolation 简介 双线性插值是一种使用重复线性插值来插值两个变量函数的方法。它通常用于以下应用&#xff1a; 图像处理和计算机视觉&…

Linux | 安装 Samba将ubuntu 的存储空间指定为windows 上的一个磁盘

01 安装 samba 文件来实现。比如把我们 ubuntu 的存储空间指定为我们 windows 上的一个磁盘,然后我们在这个磁盘里面创建 .c 文件,进行我们代码的修改和编写,可以安装 samba 文件来实现。 samba 是一种网络共享服务,可以通过网络访问我们指定的文件夹 02 第一步:下…

一文说清预训练与微调:AI的双重训练法则

什么是预训练&#xff1f; 预训练是大型语言模型训练的第一步。它在资金和计算能力的支持下&#xff0c;通过深入分析大量的文本数据&#xff0c;使模型建立起语言的基本构架。在这一阶段&#xff0c;模型通过学习海量的书籍、文章和网页&#xff0c;识别出语言的语法、句法和…

996引擎-接口测试:音效测试NPC

996引擎-接口测试:音效测试NPC 参考资料local offset = 1 -- 默认偏移量function main(player, newOffset)offset = newOffset or offset -- 更新偏移量local buttonWidth =

miniconda安装保姆级教程|win11|深度学习环境配置

一、官网安装miniconda miniconda官网&#xff1a;Miniconda - Anaconda 点击Download按钮 在红框位置输入邮箱并点击submit&#xff0c;下载链接将会发到邮箱中 邮箱中将会收到如图所示邮件&#xff0c;点击下载 选择windows对应的miniconda安装包 miniconda安装包安装完成如…

Linux shell脚本-概述、语法定义、自定义变量、环境变量、预设变量、变量的特殊用法(转义字符、单双引号、大小括号)的验证

目录 1.shell概述 1.1作为应用程序&#xff1a; 1.2 shell 作为一门语言 2.shell 语法 2.1 shell脚本的定义与执行 &#xff08;1&#xff09;新建文件 &#xff08;2&#xff09;程序开头第一行 必须写shell的类型 &#xff08;3&#xff09;程序编写完后&#xff0c…

数据驱动进化:AI Agent如何重构手机交互范式?

如果说AIGC拉开了内容生成的序幕&#xff0c;那么AI Agent则标志着AI从“工具”向“助手”的跨越式进化。它不再是简单的问答机器&#xff0c;而是一个能够感知环境、规划任务并自主执行的智能体&#xff0c;更像是虚拟世界中的“全能员工”。 正如行业所热议的&#xff1a;“大…

DL学习笔记:穿戴设备上的轻量级人体活动识别方法

Hello&#xff0c;大家好&#xff01;这里是《Dream 的深度学习笔记》,本系列将聚焦三个学习方面&#xff1a; 论文解读&#xff1a;拆解经典论文与最新突破 技术实现&#xff1a;从模型搭建到实际部署 应用案例&#xff1a;涵盖图像识别、深度学习、人工智能等热门方向 让…

windows安装配置FFmpeg教程

1.先访问官网&#xff1a;https://www.gyan.dev/ffmpeg/builds/ 2.选择安装包Windows builds from gyan.dev 3. 下滑找到release bulids部分&#xff0c;选择ffmpeg-7.0.2-essentials_build.zip 4. 然后解压将bin目录添加path系统变量&#xff1a;\ffmpeg-7.0.2-essentials_bui…

Qt/C++项目积累:4.远程升级工具 - 4.1 项目设想

背景&#xff1a; 桌面程序一般都支持远程升级&#xff0c;也是比较常用的场景设计。如酷狗音乐的升级&#xff0c;会提供两个选项&#xff0c;自动帮助安装或是新版本提醒&#xff0c;由用户来决定是否升级&#xff0c;都属于远程升级的应用及策略。 看看经过这块的功能了解及…

同旺科技USB to SPI 适配器 ---- 指令循环发送功能

所需设备&#xff1a; 内附链接 1、同旺科技USB to SPI 适配器 1、周期性的指令一次输入&#xff0c;即可以使用 “单次发送” 功能&#xff0c;也可以使用 “循环发送” 功能&#xff0c;大大减轻发送指令的编辑效率&#xff1b; 2、 “单次发送” 功能&#xff0c;“发送数据…

单表达式倒计时工具:datetime的极度优雅(Kimi)

一个简单表达式&#xff0c;也可以优雅自成工具。 笔记模板由python脚本于2025-03-22 20:25:49创建&#xff0c;本篇笔记适合任意喜欢学习的coder翻阅。 【学习的细节是欢悦的历程】 博客的核心价值&#xff1a;在于输出思考与经验&#xff0c;而不仅仅是知识的简单复述。 Pyth…

Linux:基础IO---文件描述符

文章目录 1. 前言1.1 C语言文件知识回顾 2. 文件2.1 文件基础知识 3. 被打开的文件3.1 以C语言为主&#xff0c;先回忆一下C文件接口3.2 过渡到系统&#xff0c;认识文件系统调用3.3 访问文件的本质3.4 重定向&&缓冲区 序&#xff1a;在深入了解了进程的内容后&#xf…

LINUX基础 [二] - 进程概念

目录 前言 什么是进程 如何管理进程 描述进程 组织进程 如何查看进程 通过 ps 命令查看进程 通过 ls / proc 命令查看进程 通过系统调用 获取进程标示符 前言 在学习了【Linux系统编程】中的 ​ 操作系统 和 冯诺依曼体系结构 之后&#xff0c;我们已经对系统应该有…

浏览器自动携带cookie注意事项

文章目录 浏览器自动携带与目标域相关的 cookie 是由 HTTP 协议规范和浏览器设计共同决定的一、Cookie 携带的基本规则同源策略下的自动携带跨域请求的受限携带一、服务器端配置二、客户端配置三、 常见错误及注意事项 二、Cookie 属性的筛选逻辑 三、浏览器携带cookie的准则1.…