61、Python之函数高级:为函数添加方法,实现属性可变的装饰器

2024-09-05 07:12

本文主要是介绍61、Python之函数高级:为函数添加方法,实现属性可变的装饰器,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

引言

今天文章的标题,初读起来可能有些拗口,什么叫“为函数添加方法”?但是,如果真正对“Python函数也是对象”这个理念有清晰的理解的话,其实,也是不难理解的,本质上就是给一个对象新增一个自定义方法。通过这样做,我们就可以实现在运行过程中,对装饰器的属性进行动态修改了。

本文的主要内容有:

1、函数对象添加自定义方法

2、属性可变的动态装饰器

函数对象添加自定义方法

其实,这一点比较简单,只需要把函数当作Python中的一个普通对象即可。但是,如果一直执着于函数本身,则理解起来,会有些拧巴。

我们来演示下给函数添加自定义方法的做法,直接看代码:

def add(a, b):base = 0if hasattr(globals()['add'], 'base'):base = globals()['add'].baseprint('执行加法运算')return base + a + bdef change_base(x):globals()['add'].base = xprint(f'变更后:add.base={x}')print(add(10, 20))
print('=' * 20)
add.change_base = change_base
add.change_base(100)
print('=' * 20)
print(add(10, 20))

执行结果:

dbbcf4d0d79da1ad4f305adc3ceb1f82.jpeg

首先要说明的,这是一个很拧巴的把函数当作普通对象进行使用的演示代码。

可以看到,我们给函数对象添加了一个change_base()的方法,用于修改add函数对象的base属性(当然,如果不存在base属性,就会动态添加一个base属性)。

属性可变的动态装饰器

既然在Python中一个函数对象可以当作普通对象使用,动态添加属性和方法都是可以的。那么,我们基于这样的特性,就可以进一步扩充作为装饰器的闭包函数对象,实现动态修改闭包属性的效果。

假设有这样一个业务需求,还是基于免费用户、普通vip、超级vip的用户的区分。免费用户转换为vip/svip门槛有些高,我们为了提高营收,增加了这样一项功能:支持免费用户按次付费,体验vip、svip的功能体验,从而增加付费用户的转化率。

这个业务需求,就需要我们在免费用户单次付费后,动态调整延迟时长了,也就是我们本文的主角:属性可变的动态装饰器。

我们稍微简化一下,之前的代码,简单定义一个延迟的装饰器,然后模拟付费体验一次的场景实现:

import timedef wait(func):paid = Falsedef wrap(*args, **kwargs):nonlocal paidif paid:print('使用专属付费渠道')paid = Falseelse:print('免费用户排队中...')time.sleep(5)return func(*args, **kwargs)def pay_once():nonlocal paidpaid = Truewrap.pay_once = pay_oncereturn wrap@wait
def business():print("开展正常业务")if __name__ == '__main__':print('付费体验前:')business()print('=' * 20)print('付费体验一次vip:')business.pay_once()business()print('=' * 20)print('付费体验一次vip用完后:')business()

执行结果:

78556ec6a56b56718e2f0a8890d81cca.jpeg

从执行结果,可以看到,已经模拟出了付费体验一次vip功能,之后立马又是免费的排队模式的场景。

值得注意的点有:

1、由于后续访问的都是闭包内部的函数对象,也就是wrap,所以,定义了pay_once()的方法,也是要绑定到wrap对象的,所以,才会有:wrap.pay_once = pay_once这行代码,不然,根本没法调用pay_once()方法。

2、这个装饰器通过有点拧巴的组合实现了一个修改闭包状态、使用闭包状态的功能的组合,wrap本身用于访问闭包中的自由变量paid,pay_once()用于修改paid。

3、在内部函数中要修改闭包中的自由变量paid,需要首先通过nonlocal关键字进行声明,否则会被Python解释器当作再内部函数定义一个局部变量paid。

从效果来看,本身就是通过闭包的使用,在模拟实现普通对象的对属性的读写操作的组合。虽然有些拧巴,但是,如果已经掌握了装饰器、闭包的概念,理解起来,也是比较简单的。

总结

本文通过实例代码再次验证了Python中的函数也是对象的理念,同时,明确了函数不是只能作为可调用对象进行函数调用的操作,也可以像普通对象一样,进行动态属性、方法的添加。最后,通过函数对象动态添加方法的机制,实现了一个属性可变的装饰器,实现更加灵活的业务场景需求。

感谢您的拨冗阅读,希望对您有些许帮助!

9bed6215651668339872cc1c48397ca4.jpeg

这篇关于61、Python之函数高级:为函数添加方法,实现属性可变的装饰器的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Python如何计算两个不同类型列表的相似度

《Python如何计算两个不同类型列表的相似度》在编程中,经常需要比较两个列表的相似度,尤其是当这两个列表包含不同类型的元素时,下面小编就来讲讲如何使用Python计算两个不同类型列表的相似度吧... 目录摘要引言数字类型相似度欧几里得距离曼哈顿距离字符串类型相似度Levenshtein距离Jaccard相

Ubuntu固定虚拟机ip地址的方法教程

《Ubuntu固定虚拟机ip地址的方法教程》本文详细介绍了如何在Ubuntu虚拟机中固定IP地址,包括检查和编辑`/etc/apt/sources.list`文件、更新网络配置文件以及使用Networ... 1、由于虚拟机网络是桥接,所以ip地址会不停地变化,接下来我们就讲述ip如何固定 2、如果apt安

Python安装时常见报错以及解决方案

《Python安装时常见报错以及解决方案》:本文主要介绍在安装Python、配置环境变量、使用pip以及运行Python脚本时常见的错误及其解决方案,文中介绍的非常详细,需要的朋友可以参考下... 目录一、安装 python 时常见报错及解决方案(一)安装包下载失败(二)权限不足二、配置环境变量时常见报错及

Python中顺序结构和循环结构示例代码

《Python中顺序结构和循环结构示例代码》:本文主要介绍Python中的条件语句和循环语句,条件语句用于根据条件执行不同的代码块,循环语句用于重复执行一段代码,文章还详细说明了range函数的使... 目录一、条件语句(1)条件语句的定义(2)条件语句的语法(a)单分支 if(b)双分支 if-else(

Go路由注册方法详解

《Go路由注册方法详解》Go语言中,http.NewServeMux()和http.HandleFunc()是两种不同的路由注册方式,前者创建独立的ServeMux实例,适合模块化和分层路由,灵活性高... 目录Go路由注册方法1. 路由注册的方式2. 路由器的独立性3. 灵活性4. 启动服务器的方式5.

Python itertools中accumulate函数用法及使用运用详细讲解

《Pythonitertools中accumulate函数用法及使用运用详细讲解》:本文主要介绍Python的itertools库中的accumulate函数,该函数可以计算累积和或通过指定函数... 目录1.1前言:1.2定义:1.3衍生用法:1.3Leetcode的实际运用:总结 1.1前言:本文将详

Java深度学习库DJL实现Python的NumPy方式

《Java深度学习库DJL实现Python的NumPy方式》本文介绍了DJL库的背景和基本功能,包括NDArray的创建、数学运算、数据获取和设置等,同时,还展示了如何使用NDArray进行数据预处理... 目录1 NDArray 的背景介绍1.1 架构2 JavaDJL使用2.1 安装DJL2.2 基本操

最长公共子序列问题的深度分析与Java实现方式

《最长公共子序列问题的深度分析与Java实现方式》本文详细介绍了最长公共子序列(LCS)问题,包括其概念、暴力解法、动态规划解法,并提供了Java代码实现,暴力解法虽然简单,但在大数据处理中效率较低,... 目录最长公共子序列问题概述问题理解与示例分析暴力解法思路与示例代码动态规划解法DP 表的构建与意义动

在不同系统间迁移Python程序的方法与教程

《在不同系统间迁移Python程序的方法与教程》本文介绍了几种将Windows上编写的Python程序迁移到Linux服务器上的方法,包括使用虚拟环境和依赖冻结、容器化技术(如Docker)、使用An... 目录使用虚拟环境和依赖冻结1. 创建虚拟环境2. 冻结依赖使用容器化技术(如 docker)1. 创

java父子线程之间实现共享传递数据

《java父子线程之间实现共享传递数据》本文介绍了Java中父子线程间共享传递数据的几种方法,包括ThreadLocal变量、并发集合和内存队列或消息队列,并提醒注意并发安全问题... 目录通过 ThreadLocal 变量共享数据通过并发集合共享数据通过内存队列或消息队列共享数据注意并发安全问题总结在 J