当前位置: 首页 > news >正文

Python魔法方法__call__深入详解

目录

1、魔法方法__call__初探 🧙‍♂️

1.1 什么是__call__?

1.2 基础用法演示

1.3 自定义行为与参数传递

2、实现轻量级装饰器模式 🎗️

2.1 装饰器概念回顾

2.2 利用__call__构建装饰器

2.3 深入理解装饰器应用场景

3、类实例变身函数调用 🔮

3.1 类似函数的行为模拟

3.2 动态执行与灵活性提升

3.3 实战案例:日志记录器

4、实现状态机模式 🔄

4.1 状态机概念回顾

4.2 通过__call__管理状态转换

示例代码

4.3 应用实例:简单自动售货机模拟

5、构建轻量级ORM模型 📡

5.1 自定义数据访问对象

示例代码

5.2 实现类的实例直接查询数据库

5.3 链式调用优化查询体验

6、高级用法:元类与__call__结合 🔍

6.1 元类回顾与作用

6.2 通过__call__自定义类实例化过程

6.3 应用实例:类的自动注册系统

6.4 利用元类动态修改__call__

7、性能优化与内存管理 🚀

7.1 __call__与缓存技术

7.2 减少重复实例化开销

7.3 实战分析:性能对比测试

8、总结与展望 🌌



1、魔法方法__call__初探 🧙‍♂️

1.1 什么是__call__?

在Python中,__call__是一个特殊方法,赋予了对象可被直接调用的能力 ,就像函数一样。当一个类实例被当作函数调用时,实际上就是在调用这个类的__call__方法。这为设计灵活、行为动态的对象提供了强大手段,使得对象可以模仿函数行为,实现更高级的面向对象编程模式。

1.2 基础用法演示

下面是一个简单的__call__使用示例,定义了一个Counter类 ,用于计数每次调用的次数:

class Counter:
    def __init__(self):
        self.count = 0    def __call__(self):
        self.count += 1
        return self.count# 创建Counter实例
my_counter = Counter()# 直接调用实例 ,就像调用函数
print(my_counter())  # 输出: 1
print(my_counter())  # 输出: 2

1.3 自定义行为与参数传递

__call__方法不仅限于无参数调用,它还可以接收任意数量的位置参数和关键字参数,从而实现更加复杂的逻辑。比如,创建一个Multiplier类,使其能够接受一个乘数并返回与该乘数相乘的结果:

class Multiplier:
    def __init__(self, factor):
        self.factor = factor    def __call__(self, value):
        return self.factor * value# 使用Multiplier类
times_three = Multiplier(3)# 传递参数调用实例
print(times_three(10))  # 输出: 30

通过这种方式,__call__魔法方法不仅增加了代码的可读性和灵活性 ,还为实现更高级的设计模式 ,如装饰器、策略模式等,奠定了基础。掌握__call__的应用,是深入理解Python面向对象编程的重要一步。

2、实现轻量级装饰器模式 🎗️

2.1 装饰器概念回顾

装饰器是一种特殊类型的函数,可以修改其他函数的功能或行为,而无需更改被修饰函数的源代码。它们在Python中广泛应用于日志记录、性能测试、权限校验等多种场景,极大地增强了代码的可重用性和灵活性。

2.2 利用__call__构建装饰器

利用类的__call__方法可以轻松实现装饰器功能,这种方式让装饰器本身更加模块化和可配置。下面是一个简单的示例,展示如何通过定义一个类作为装饰器来记录函数的执行时间:

import timeclass TimerDecorator:
    def __init__(self, func):
        self.func = func    def __call__(self, *args, **kwargs):
        start_time = time.time()
        result = self.func(*args, **kwargs)
        end_time = time.time()
        print(f"{self.func.__name__} executed in {end_time - start_time:.4f}s")
        return result@TimerDecorator
def example_function():
    time.sleep(1)
    print("Function executed")example_function()

在这个例子中,TimerDecorator类通过__call__方法实现了装饰器逻辑 ,测量并打印了被装饰函数example_function的执行时间。

2.3 深入理解装饰器应用场景

装饰器的使用远不止于此,它在实际开发中扮演着多面手的角色:

  • • 日志记录:自动记录函数调用的日志,包括入参、出参及异常信息 ,便于监控和调试。

  • • 性能测试:评估函数执行效率,如上述时间记录装饰器。

  • • 权限控制:在函数执行前检查用户权限,增强安全性。

  • • 缓存机制:对耗时操作的结果进行缓存,提高程序响应速度。

  • • 参数验证:自动验证函数输入参数的有效性,减少错误处理代码。

通过结合__call__,我们可以创建出更加复杂和灵活的装饰器,为Python程序添加丰富的功能,同时保持代码的整洁和可维护性。

3、类实例变身函数调用 🔮

3.1 类似函数的行为模拟

通过实现__call__方法,类实例可以像普通函数那样直接被调用。这种设计模式允许我们封装复杂的逻辑和状态到类中,同时保持调用接口的简洁。例如,创建一个MathOperation类 ,其行为如同一个加法函数,但内部可以包含更复杂的计算逻辑:

class MathOperation:
    def __init__(self, a, b):
        self.a = a
        self.b = b    def __call__(self):
        return self.a + self.b# 实例化并像函数一样调用
addition = MathOperation(3, 4)
result = addition()  # 输出: 7

3.2 动态执行与灵活性提升

__call__方法的动态特性,使得类可以根据运行时的情况调整行为。这对于需要在调用时刻决定具体执行逻辑的场景非常有用。例如,一个根据用户输入动态选择算法的框架:

class AlgorithmSelector:
    def __init__(self, algorithm_name):
        self.algorithm_name = algorithm_name    def __call__(self, data):
        if self.algorithm_name == 'sort':
            return sorted(data)
        elif self.algorithm_name == 'reverse':
            return data[::-1]
        else:
            raise ValueError("Unsupported algorithm")# 动态选择排序或反转操作
selector = AlgorithmSelector('sort')
sorted_list = selector([3, 1, 4, 1, 5])  # 输出: [1, 1, 3, 4, 5]selector = AlgorithmSelector('reverse')
reversed_list = selector([3, 1, 4, 1, 5])  # 输出: [5, 1, 4, 1, 3]

3.3 实战案例:日志记录器

构建一个简单的日志记录器,利用__call__方法实现不同级别的日志输出,并能够动

相关文章:

Python魔法方法__call__深入详解

目录 1、魔法方法__call__初探 🧙‍♂️ 1.1 什么是__call__? 1.2 基础用法演示 1.3 自定义行为与参数传递 2、实现轻量级装饰器模式 🎗️ 2.1 装饰器概念回顾 2.2 利用__call__构建装饰器 2.3 深入理解装饰器应用场景 3、类实例变身函数调用 🔮 3.1 类似函数的…...

PyQt5 生成py文件不能运行;pushButton点击事件;QTextEdit 获取输入框内容

目录 cant open file c.pyuic: c.pyuic $FileName$ -o $FileNameWithoutExtension$.p PyQt5 生成py文件不能运行 pushButton点击事件 QTextEdit 获取输入框内容 整体运行代码: Creating a Qt Widget Based Application | Qt Creator Manual cant open file c.pyuic: c.…...

HarmonyOS最佳实践文档总结汇总(面试题可能会问)

api12 上面来了最佳实现方案,未来面试题有的问了 编号分类内容子类链接 1性能体验设计体验设计概述 文档中心用户体验设计 文档中心流畅评测指标 文档中心交互流畅体验设计 文档中心视觉流畅体验设计 文档中心2性能优化开发高性能ArkUIUI组件性能优化文档中心合…...

leetcode 56合并区间

思路 合并就是首先应该按照left左边界排序,排完序以后,如果i的左边界小于等于i-1的右边界,说明有重合,此时这两个可以合并,右边界应该取最大值。 代码 排序 我是定义了一个类,存储左右边界,先将数组转化…...

企业微信内嵌H5项目接入聊天功能

产品需求是,在列表中把符合条件的列表接入聊天功能,以下是详细步骤: 1.引入企业微信 <script src"https://res.wx.qq.com/wwopen/js/jsapi/jweixin-1.0.0.js"></script> 2.获取wx签名(必须要) /*** 获取wx签名**/ export function getWxJsApi(data) {r…...

微信小程序 this.setData高级用法(只更改单个数据)

合理使用 setData | 微信开放文档 1、页面 <view class"h-100px"></view> <view>最简单的数据&#xff1a;</view> <button bind:tap"handleAdd" data-type"1">点我加 1&#xff1a; {{text}}</button> &…...

使用npm发布自己的插件包

文章目录 1. 准备工作1.1 拥有一个npm账号1.2 准备你的插件代码1.3 编写package.json文件 2. 本地测试3. 发布到npm3.1 登录npm3.2 发布插件3.3 更新插件 4. 注意事项 在JavaScript和Node.js的生态系统中&#xff0c;npm&#xff08;Node Package Manager&#xff09;是一个非常…...

前端入门篇(五十二)练习6:transition过渡小动画

所以应该先找到第n个li&#xff0c;找到li再找img&#xff0c;li没有找错&#xff0c;底下又各自只有一个img&#xff0c;解决 ul li:nth-child(1) img { } 描述文字从下往上&#xff1a; 一开始描述也在框框下面&#xff0c;当hover时&#xff0c;translateY(0)&#xff0…...

scrapy模块的基础使用

scrapy模块是爬虫工作者最常用的一个模块之一&#xff0c;因它有许多好用的模板&#xff0c;和丰富的中间件&#xff0c;深受欢迎。 一&#xff0c;scrapy的安装 可以通过pypi的指引进行安装 在终端内输入以下代码&#xff1a; pip install scrapy 二&#xff0c;项目的建…...

如何在不降低网络安全防护的前提下,优化pcdn的流量清洗效率?

在不降低网络安全防护的前提下&#xff0c;优化PCDN的流量清洗效率是一个复杂但至关重要的任务。以下是一些建议&#xff0c;帮助您实现这一目标&#xff1a; 一&#xff0e;升级硬件与网络设备&#xff1a; 投资于高性能的硬件和网络设备&#xff0c;以确保流量清洗过程中的…...

linux发行版CentOS、Debian和Ubuntu的对比

一、CentOS、Debian和Ubuntu优缺点比较 CentOS、Debian和Ubuntu是目前国内云服务市场上最常见三个linux发行版本&#xff0c;在我们选购云服务时&#xff0c;要怎么选择&#xff1f;以下表格详细介绍了三者之间的优缺点和适用场景。 特性CentOSDebianUbuntu优点稳定性高&…...

WordPress如何删除内存中的缓存?

今天boke112百科将某篇文章修改分类和内容更新后&#xff0c;发现文章底部的相关文章显示的内容跟文章分类、标签毫无关系&#xff0c;还是显示原来的旧内容。后来查看YIA主题相关文章的代码&#xff0c;才发现相关文章的数据保存到内存中的&#xff0c;而且是永不过期&#xf…...

【XML模版文件参数初始化】

XML 模版文件&#xff0c;内部存在需要自定义的数据&#xff0c;使用 Python 进行初始化。 1、存在一个 XML 模版文件&#xff0c;定义如下 <!-- 文件名称 index.xml --> <root><HEAD><VER>1.0</VER><SRC>10000000000000</SRC><…...

Golang | Leetcode Golang题解之第160题相交链表

题目&#xff1a; 题解&#xff1a; func getIntersectionNode(headA, headB *ListNode) *ListNode {if headA nil || headB nil {return nil}pa, pb : headA, headBfor pa ! pb {if pa nil {pa headB} else {pa pa.Next}if pb nil {pb headA} else {pb pb.Next}}retu…...

基于FOC控制器的BLDC无刷直流电机控制系统matlab编程与仿真

目录 1.课题概述 2.系统仿真结果 3.核心程序与模型 4.系统原理简介 5.完整工程文件 1.课题概述 基于FOC控制器的BLDC无刷直流电机控制系统matlab编程与仿真&#xff0c;使用MATLAB编程实现&#xff0c;包括FOC控制器&#xff0c;clark&#xff0c;park等&#xff0c;不使用…...

ffmpeg转换视频格式

ffmpeg -i "录屏 2024-06-16 01-56-40.webm" -vf "scale1912:1070" -vcodec libx264 pit.mp4如果你觉得视频压缩速度太慢&#xff0c;可以尝试以下几种方法来加速视频处理&#xff1a; 1. 使用多线程 FFmpeg 支持多线程&#xff0c;可以利用多个 CPU 核心…...

设计程序,利用栈实现数值转换

二、利用栈实现数值转换&#xff0c;先定义一个栈的顺序存储结构&#xff0c;那么我们需要定义一个结构体&#xff0c;结构体里面有个int类型的数组&#xff0c;还有一个top用来存储栈顶元素的下标。栈是一种基本的数据结构&#xff0c;它遵循先进后出的原则。这意味着最后添加…...

QSharedMemory使用详解

QSharedMemory 是 Qt 提供的一个类&#xff0c;用于在多个进程之间共享内存。它可以让您在不同的进程间传递数据&#xff0c;而无需通过文件或网络来进行传输。下面是 QSharedMemory 的详细用法和相关知识点。 一、基本概念 共享内存&#xff1a;共享内存是一块可以被多个进程…...

中电金信:保险业多项举措共绘数字化转型新篇章

...

Gartner发布2024年人工智能技术成熟度曲线:29项决定人工智能领域发展方向的前沿和趋势性技术

人工智能投资已达到新高&#xff0c;重点是生成式人工智能&#xff0c;但在大多数情况下&#xff0c;该技术尚未实现预期的商业价值。这项研究通过分析各种人工智能创新&#xff08;其中许多创新正在快速发展&#xff09;&#xff0c;帮助人工智能领导者确定其他值得投资的技术…...

2024年赣州旅游投资集团社会招聘笔试真

2024年赣州旅游投资集团社会招聘笔试真 题 ( 满 分 1 0 0 分 时 间 1 2 0 分 钟 ) 一、单选题(每题只有一个正确答案,答错、不答或多答均不得分) 1.纪要的特点不包括()。 A.概括重点 B.指导传达 C. 客观纪实 D.有言必录 【答案】: D 2.1864年,()预言了电磁波的存在,并指出…...

电脑插入多块移动硬盘后经常出现卡顿和蓝屏

当电脑在插入多块移动硬盘后频繁出现卡顿和蓝屏问题时&#xff0c;可能涉及硬件资源冲突、驱动兼容性、供电不足或系统设置等多方面原因。以下是逐步排查和解决方案&#xff1a; 1. 检查电源供电问题 问题原因&#xff1a;多块移动硬盘同时运行可能导致USB接口供电不足&#x…...

【项目实战】通过多模态+LangGraph实现PPT生成助手

PPT自动生成系统 基于LangGraph的PPT自动生成系统&#xff0c;可以将Markdown文档自动转换为PPT演示文稿。 功能特点 Markdown解析&#xff1a;自动解析Markdown文档结构PPT模板分析&#xff1a;分析PPT模板的布局和风格智能布局决策&#xff1a;匹配内容与合适的PPT布局自动…...

UR 协作机器人「三剑客」:精密轻量担当(UR7e)、全能协作主力(UR12e)、重型任务专家(UR15)

UR协作机器人正以其卓越性能在现代制造业自动化中扮演重要角色。UR7e、UR12e和UR15通过创新技术和精准设计满足了不同行业的多样化需求。其中&#xff0c;UR15以其速度、精度及人工智能准备能力成为自动化领域的重要突破。UR7e和UR12e则在负载规格和市场定位上不断优化&#xf…...

ArcGIS Pro制作水平横向图例+多级标注

今天介绍下载ArcGIS Pro中如何设置水平横向图例。 之前我们介绍了ArcGIS的横向图例制作&#xff1a;ArcGIS横向、多列图例、顺序重排、符号居中、批量更改图例符号等等&#xff08;ArcGIS出图图例8大技巧&#xff09;&#xff0c;那这次我们看看ArcGIS Pro如何更加快捷的操作。…...

Spring是如何解决Bean的循环依赖:三级缓存机制

1、什么是 Bean 的循环依赖 在 Spring框架中,Bean 的循环依赖是指多个 Bean 之间‌互相持有对方引用‌,形成闭环依赖关系的现象。 多个 Bean 的依赖关系构成环形链路,例如: 双向依赖:Bean A 依赖 Bean B,同时 Bean B 也依赖 Bean A(A↔B)。链条循环: Bean A → Bean…...

音视频——I2S 协议详解

I2S 协议详解 I2S (Inter-IC Sound) 协议是一种串行总线协议&#xff0c;专门用于在数字音频设备之间传输数字音频数据。它由飞利浦&#xff08;Philips&#xff09;公司开发&#xff0c;以其简单、高效和广泛的兼容性而闻名。 1. 信号线 I2S 协议通常使用三根或四根信号线&a…...

Kubernetes 网络模型深度解析:Pod IP 与 Service 的负载均衡机制,Service到底是什么?

Pod IP 的本质与特性 Pod IP 的定位 纯端点地址&#xff1a;Pod IP 是分配给 Pod 网络命名空间的真实 IP 地址&#xff08;如 10.244.1.2&#xff09;无特殊名称&#xff1a;在 Kubernetes 中&#xff0c;它通常被称为 “Pod IP” 或 “容器 IP”生命周期&#xff1a;与 Pod …...

深入浅出Diffusion模型:从原理到实践的全方位教程

I. 引言&#xff1a;生成式AI的黎明 – Diffusion模型是什么&#xff1f; 近年来&#xff0c;生成式人工智能&#xff08;Generative AI&#xff09;领域取得了爆炸性的进展&#xff0c;模型能够根据简单的文本提示创作出逼真的图像、连贯的文本&#xff0c;乃至更多令人惊叹的…...

32单片机——基本定时器

STM32F103有众多的定时器&#xff0c;其中包括2个基本定时器&#xff08;TIM6和TIM7&#xff09;、4个通用定时器&#xff08;TIM2~TIM5&#xff09;、2个高级控制定时器&#xff08;TIM1和TIM8&#xff09;&#xff0c;这些定时器彼此完全独立&#xff0c;不共享任何资源 1、定…...