百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术文章 > 正文

为什么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语句是按固定顺序执行的,先执行前面的语句,再执行后面的语句。如果你像要程序按照你自己定制的流程执行,就需要用到流程控制的语句,最主要用到的是条件语句和循环语句。条件语句...