为什么python高手都爱用闭包?这个实时函数技巧绝了
itomcoil 2025-07-23 15:15 4 浏览
杂谈
我想很多人都玩过python的闭包,其中最有趣的部分应该就是装饰器了。
但我想很多人应该没运用上闭包的特性——外部局部变量的存储。
什么意思呢?其实就是当闭包引用外部的局部变量将会被存储起来,而不会随着函数的结束而释放。
def test_block():
test_list = []
def inner():
test_list.append(1)
print(test_list)
return inner
test = test_block()
test() # [1]
test() # [1, 1]
test() # [1, 1, 1]
可以看到,代码中的test_list没有被重新设置为空的列表,而是在闭包中不断添加新的元素。
当被inner()闭包函数捕获以后,test_list就不是局部变量了,它会被存储起来。
今天我将基于这个特性写一个比较有趣的代码,可以随时更新实时函数,只要你传入一个.py文件就可以在任何时候更新函数,就算是程序已经在运行期间也可以。
通过这个方式很容易实现架构风格中的基于规则的架构风格,例如当我们存在一笔消费,正常情况下,满100减5,当发生活动时,需要满100减20,且该规则没有在前期开发中进行实现,那么我们在不进行程序重启的情况下是很难实现规则的改变,因此可以通过本文来实现类似的需求。
一、调用外部函数
我们先来创建一个文件handle.py,里面创建一个函数show()将在后续用到:
# handle.py
def show(name):
print(f'我叫 {name}')
然后我们来写一个代码,通过文件的路径来调用它的方法:
import importlib
spec = importlib.util.spec_from_file_location(
name='handle',
location='handle.py'
)
module = importlib.util.module_from_spec(spec)
spec.loader.exec_module(module)
module.show('小明')
我们用importlib来调用这个文件的show()函数,我们来看看输出情况:
我叫 小明
成功,我们现在只需要.py的文件即可完成调用,完成了最基本的逻辑了。
如果是其他路径模块调用,需要该模块项目中其他的方法或者类,需要将模块的路径加到系统路径下:sys.path.append('xxxxx')
二、逻辑实现
整个代码的逻辑是这样的,通过闭包来记录.py的文件路径,然后每次调用去判断这个文件的修改时间,如果与我们内存中的不一致,说明文件就被改动了,重新加载一次模块即可。
import os
import importlib
import sys
import time
def call(py_file=''):
# 判断文件是否存在
if not os.path.exists(py_file):
raise Exception('Py file does not exist')
# 文件后缀是否正确
if not py_file.endswith('py') and not py_file.endswith('pyc'):
raise Exception('Py file type error')
# 将模块加入到路径中
sys.path.append(py_file)
# 获取咱们的模块名称
module_name = py_file.split(os.path.sep)[-1]
if len(module_name.split('.')) > 0:
module_name = module_name.split('.')[0]
spec = importlib.util.spec_from_file_location(
name=module_name,
location=py_file
)
# 模块导入
module = importlib.util.module_from_spec(spec)
# 模块可以调用
spec.loader.exec_module(module)
# 用来获取module及判断它的文件修改时间
module_dict = {
'module': module,
'update_time': time.strftime('%Y%m%d%H%M%S', time.localtime(os.stat(py_file).st_mtime))
}
def inner(*args, **kwargs):
try:
update_time_str = time.strftime('%Y%m%d%H%M%S', time.localtime(os.stat(py_file).st_mtime))
except Exception as e:
raise e
# 当前文件修改时间改变,则我们模块需要重新加入
if module_dict.get('update_time') != update_time_str:
spec.loader.exec_module(module_dict['module'])
module_dict["update_time"] = update_time_str
# 外部模块的函数
return module_dict['module'].show(*args, **kwargs)
return inner
我们设定了一个局部变量module_dict,用于存储模块的信息,由于它被闭包引用了,因此变成了一个长期变量。
当我们下一次调用时,我们会判断上一次的文件修改时间和本次的区别,如果不同了,则表示模块已经发生改变,我们就做spec.loader.exec_module(module_dict['module'])的操作。
三、测试代码
我们每两秒就跑一次代码,看看执行效果:
f = call(r'xxx\xxx\xxx\handle.py')
while True:
f('小明')
time.sleep(2)
然后在中途将调用模块的函数内容改为:
def show(name):
print(f'我的名字叫 {name}')
测试结果gif:
测试结果
四、结尾
闭包是个比较神奇的概念,很多编程语言都有它的存在,在python中很多人可能只是为了实现装饰器而使用闭包,其实如果开发它的特性能够完成很多有趣的事情。
如果你用闭包玩过什么好玩的,欢迎来留言区留言!
相关推荐
- Python自动化——pytest常用插件详解
-
前言Pytest是Python的一种单元测试框架,与unittest相比,使用起来更简洁、效率更高,也是目前大部分使用python编写测试用例的小伙伴们的第一选择了。除了框架本身提供的功能外,Pyte...
- 全网最全pytest大型攻略,单元测试学这就够了
-
pytest是一款以python为开发语言的第三方测试,主要特点如下:比自带的unittest更简洁高效,兼容unittest框架支持参数化可以更精确的控制要测试的测试用例丰富的插件,已有30...
- Python Logging 最佳实践(python logging配置)
-
Pythonlogging的“最佳实践”可以概括为一句话:让日志既能在开发时帮你排错,也能在生产里帮你定位问题,同时不给运维埋坑。下面给出一份可直接落地的checklist,分场景逐条说明。1....
- Python单元测试框架对比(python中unittest框架)
-
一、核心框架对比特性unittest(标准库)pytest(主流第三方)nose2(unittest扩展)doctest(文档测试)安装Python标准库pipinstallpytestp...
- 如何使用Python进行单元测试(pycharm单元测试)
-
前言在我的日常工作中,我是一名专业程序员。我使用c++、c#和Javascript。我是一个开发团队的一员,他们使用单元测试来验证我们的代码是否按照它应该的方式工作。在本文中,我将通过讨论以下主题来研...
- Python单元测试(pycharm单元测试)
-
1.单元测试概述1.1什么是单元测试单元测试(UnitTesting)是指对软件中的最小可测试单元进行检查和验证的过程。在Python中,最小单元通常指函数、方法或类。1.2单元测试的特性独立...
- pytest框架之fixture测试夹具详解
-
前言大家晚上好呀,今天呢来和大家唠唠pytest中的fixtures夹具的详解,废话就不多说了咱们直接进入主题哈。一、fixture的优势pytest框架的fixture测试夹具就相当于unitte...
- Pytest精髓Fixture功能实例!测试效率暴涨!
-
前言大家好!我们今天来学习Python测试框架中的最具特色的功能之一:Fixture。可以说,掌握了Fixture,你就掌握了Pytest的精髓。它不仅能让你的测试代码更简洁、更优雅、更易于...
- Python最常见的170道面试题全解析答案(二)
-
60.请写一个Python逻辑,计算一个文件中的大写字母数量答:withopen(‘A.txt’)asfs:count=0foriinfs.read():ifi.isupper...
- 为什么python高手都爱用闭包?这个实时函数技巧绝了
-
杂谈我想很多人都玩过python的闭包,其中最有趣的部分应该就是装饰器了。但我想很多人应该没运用上闭包的特性——外部局部变量的存储。什么意思呢?其实就是当闭包引用外部的局部变量将会被存储起来,而不会随...
- 春节停车难?用Python找空车位(用python编写停车场停车收费)
-
【导语】今天这篇文章的选题非常贴近生活。营长生活在北京,深知开车出门最怕的就是堵车和找不到停车位。记得冬至那个周末,几个小伙伴滑雪回来找了一家饺子馆吃饺子,结果七拐八拐,好不容易才找到一个停车位。看到...
- PYTHON数据分析必备知识(2)(python数据分析范例)
-
1.二分钟快速给项目添加日志信息"""给项目添加日志信息"""#导Python内置包importloggingimporttime...
- 春节回家!车位难求啊!看我用Python自动寻找空车位!
-
作者通过相机结合深度学习算法,基于Python语言建立一个高精度的停车位的通知系统,每当有新停车位时就会发短信提醒我。听起来好像很复杂,真的方便实用吗?但实际上所使用的工具都是现成的,只要将这些工...
- “==”和“is”有什么区别?一个问题就能暴露你的Python水平
-
可能在网上你经常能看到关于这个问题的答案和解析,但是依然有很多刚开始学习Python的人,不了解这个问题,也不知道为什么问这个问题时会暴露自己是“菜鸟”,这个问题就是:“==”和“is”之间有什么...
- Python条件语句怎么用(python中条件语句的用法)
-
if条件判断语句python语句是按固定顺序执行的,先执行前面的语句,再执行后面的语句。如果你像要程序按照你自己定制的流程执行,就需要用到流程控制的语句,最主要用到的是条件语句和循环语句。条件语句...
- 一周热门
- 最近发表
- 标签列表
-
- ps图案在哪里 (33)
- super().__init__ (33)
- python 获取日期 (34)
- 0xa (36)
- super().__init__()详解 (33)
- python安装包在哪里找 (33)
- linux查看python版本信息 (35)
- python怎么改成中文 (35)
- php文件怎么在浏览器运行 (33)
- eval在python中的意思 (33)
- python安装opencv库 (35)
- python div (34)
- sticky css (33)
- python中random.randint()函数 (34)
- python去掉字符串中的指定字符 (33)
- python入门经典100题 (34)
- anaconda安装路径 (34)
- yield和return的区别 (33)
- 1到10的阶乘之和是多少 (35)
- python安装sklearn库 (33)
- dom和bom区别 (33)
- js 替换指定位置的字符 (33)
- python判断元素是否存在 (33)
- sorted key (33)
- shutil.copy() (33)