Python的反射机制
本篇文章讨论Python中非常有趣且强大的概念——反射(Reflection)。想象一下,你正在编写一段代码,并希望这段代码能够具备自我认知和动态调整的能力。就好比一面镜子,能反映出它自身的属性和行为。在编程领域,这种“自我观察”和“动态操控”的能力就是我们所说的反射。
目录
一、概念
二、具体表现
1、类型信息获取:
2、动态属性操作:
3、模块导入与查找:
三、反射的应用场景举例
四、总结与拓展
一、概念
在Python中,反射是指程序能够在运行时获取并操作自身内部信息,包括但不限于类、对象、属性和方法等。这意味着我们可以根据字符串动态地访问和修改这些元素,而不必在编译时就明确知道所有的细节。
说得通俗一点就是通过指定的字符串(即关键字),从而获取已存在的对象中的方法或属性。找到方法后自动执行——基于字符串的事件驱动;反射在自动化测试中有非常广泛的应用,是关键字驱动框架的基础。
二、具体表现
1、类型信息获取:
type(obj):返回对象obj的类型。
isinstance(obj, classinfo):判断对象obj是否是类或类型classinfo的实例。
2、动态属性操作:
- getattr(obj, attr_name, default) : 通过字符串
attr_name从对象obj中获取属性或方法。如果该属性不存在且提供了默认值,则返回默认值;否则如果没有提供默认值且属性不存在,将抛出AttributeError异常。- hasattr(obj, attr_name) :检查对象obj是否有名为attr_name的属性或方法。
- setattr(obj, attr_name, value) :为对象obj设置名为attr_name的属性或方法,将其值设为value。
- delattr(obj, attr_name) :删除对象obj的指定属性或方法attr_name。
- dir(obj) : 返回一个包含对象
obj所有属性名和方法名的列表,这是了解对象可访问成员的一个便捷方式。
通过字符串的形式操作对象的属性或方法这一过程确实体现了反射机制。这里是一个简单的例子说明如何使用字符串动态地访问和调用对象的属性与方法:
class MyClass:def __init__(self):self.my_attribute = "Hello, World!"def my_method(self):return "Called my_method()"# 创建一个对象
obj = MyClass()#使用dir(obj)打印对象的属性
print(dir(obj))# 字符串形式的属性名
attr_name = 'my_attribute'
method_name = 'my_method'
my_name='lalala'# 通过getattr获取并打印属性值
attribute_value = getattr(obj, attr_name)
print(attribute_value) # 输出: Hello, World!
attrdefault_value=getattr(obj,my_name,'没有该属性返回默认值')
print(attrdefault_value) # 输出: 没有该属性返回默认值# 通过getattr调用方法,并打印结果
method = getattr(obj, method_name)
result = method()
print(result) # 输出: Called my_method()# 通过setattr设置属性值
new_attr_name = 'new_attribute'
new_attribute_value = "New value"
setattr(obj, new_attr_name, new_attribute_value)
print(getattr(obj, new_attr_name)) # 输出: New value# 使用delattr删除属性
delattr(obj, new_attr_name)
print(hasattr(obj, new_attr_name)) # 输出: False
3、模块导入与查找:
动态导入模块:使用importlib.import_module(name)函数根据字符串名称导入模块。
在已导入的模块中,可以通过字符串动态查找和调用函数、类或变量。下面是一个具体的例子:
# 导入importlib模块
import importlib# 假设我们要动态导入一个名为'my_module'的模块,其路径在当前环境的PYTHONPATH中
module_name = "my_module"# 动态导入模块
module = importlib.import_module(module_name)# 现在我们可以访问该模块中的任何全局属性,如函数、类或变量
# 假设该模块有一个名为'some_function'的函数
function_name = "some_function"# 动态获取并调用函数
if hasattr(module, function_name):the_function = getattr(module, function_name)the_function()# 同样,也可以查找和实例化模块中的类
class_name = "MyClass"
if hasattr(module, class_name):MyClass = getattr(module, class_name)instance = MyClass()instance.some_method()
三、反射的应用场景举例
需求描述:用户通过输入字符串来调用对象的对应方法,通过模拟一个服务器响应用户的请求,设置有注册页、登录页、主页、关于页以及错误页。
class WebSite:def register(self):print("欢迎来到注册页面")def login(self):print("欢迎来到登录页面")def home(self):print("欢迎进入主页")def about(self):print("关于我们")def error(self):print("404 No Found!")web=WebSite()
while True:choce=input("请输入要访问的页面:")if choce=="register":web.register()elif choce=="login":web.login()elif choce=="home":web.home()elif choce=="about":web.about()elif choce=="error":web.error()else:print("请输入正确的页面")
这样看实现了一个简单的网站功能,但由于代码段对用户的请求页判断的代码块冗长,当新增一个网页时也要实时修改对应的主体代码,维护起来不方便。这时可以利用反射来动态操作:
while True:choce=input('请输入要访问的页面:')if hasattr(web,choce):getattr(web,choce)()else:web.error()
无论新增多少网页和操作,反射都能轻松解决。
四、总结与拓展
Python反射机制允许程序在运行时访问、检查和修改其内部结构和行为。它主要通过内建函数getattr(), setattr(), delattr(),以及hasattr()等实现对类或对象属性、方法的动态操作。此外,dir()函数可以帮助我们查看一个对象的所有属性和方法名,进一步揭示了对象的内在结构。
具体应用场景包括但不限于:
- 动态调用方法:根据字符串参数来决定执行哪个方法,这在Web框架中的路由处理、插件系统设计以及用户自定义命令执行等方面非常实用。
- 自动化测试与调试:通过反射可以编写更为灵活的测试脚本,无需预先知道所有类和对象的方法及属性名。
- 序列化与反序列化:在数据转换过程中,反射能够帮助遍历对象的所有属性进行序列化或反序列化。
- 配置驱动编程:根据配置文件加载不同的类或模块,利用反射创建并初始化对象。
- 元编程与内省:通过反射可以编写更加智能和适应性强的代码。
拓展:
- 虽然反射提供了极大的灵活性,但它也存在一些潜在的问题:
- 可读性降低:过度使用反射可能导致代码难以理解和维护,因为相关的操作不是直接明确地写出来,而是隐含在字符串变量或其他动态信息中。
- 安全隐患:如果来自不受信任的源的数据用于反射操作,可能会导致意外的调用和安全漏洞。
- 性能影响:相较于静态解析和编译,反射通常涉及额外的查找和解释过程,可能对性能产生一定影响。
希望通过这篇文章,大家不仅了解了Python反射的基本原理和应用场景,也能意识到何时以及如何合理利用这一特性来提高代码的可扩展性和复用性。
相关文章:
Python的反射机制
本篇文章讨论Python中非常有趣且强大的概念——反射(Reflection)。想象一下,你正在编写一段代码,并希望这段代码能够具备自我认知和动态调整的能力。就好比一面镜子,能反映出它自身的属性和行为。在编程领域࿰…...
Python数学建模-2.9Matplotlib库
Matplotlib库是Python中一个非常流行的绘图库,它提供了大量的绘图工具,可以生成各种类型的静态、动态、交互式的图表。Matplotlib的设计初衷是为了与NumPy配合使用,从而提供一个强大的数学绘图工具。 1.Matplotlib的主要特点 丰富的图表类型…...
【MySQL】数据库的基础概念
👦个人主页:Weraphael ✍🏻作者简介:目前学习计网、mysql和算法 ✈️专栏:MySQL学习 🐋 希望大家多多支持,咱一起进步!😁 如果文章对你有帮助的话 欢迎 评论💬…...
Linux:离线安装 jdk-8(配置Java环境)
Linux:离线安装 jdk-8(配置Java环境) 1、jdk简介2、检查已安装的Java版本,并卸载3、准备安装包4、解压安装包、进行安装5、设置环境变量(全局/个人) 💖The Begin💖点点关注,收藏不迷路…...
【DP】第十三届蓝桥杯省赛C++ B组《李白打酒加强版》(C++)
【题目描述】 话说大诗人李白,一生好饮。 幸好他从不开车。 一天,他提着酒壶,从家里出来,酒壶中有酒 2 斗。 他边走边唱: 无事街上走,提壶去打酒。 逢店加一倍,遇花喝一斗。 这一路上&am…...
数据结构试卷第九套
1.时间复杂度 2.树,森林,二叉树的转换 2.1树转二叉树 给所有的兄弟节点之间加一条连线;去线,只保留当前根节点与第一个叶子节点的连线,删除它与其他节点之间的连线;然后根据左孩子右兄弟进行调整…...
【Linux第三课-基础开发工具的使用】yum、vim、gcc/g++编译器、gdb、Make/Makefile编写、进度条程序、git命令行简单操作
目录 yum - 软件包管理器快速认识yum快速使用yumyum搜索yum安装yum卸载 yum的周边 - yum的整个生态问题 vim快速介绍vimvim的模式命令模式插入模式低行模式 常见模式 -- 命令、低行命令模式 -- 光标的移动命令模式 -- 复制粘贴、剪贴、删除命令模式 -- 小写/大写替换模式命令模…...
Redis:ClassCastException【bug】
Redis:ClassCastException【bug】 前言版权Redis:ClassCastException【bug】错误产生相关资源控制器:UserController("/user")配置:RedisConfiguration实体类:User数据表:User 解决 最后 前言 2…...
JSON 配置文件
JSON 配置文件的作用 JSON 是一种数据格式,在实际开发中, JSON 总是以配置文件的形式出现。小程序项目中也不例外:通过不同的 .json 配置文件,可以对小程序项目进行不同级别的配置。 小程序项目中有 4 种 json 配置文件࿰…...
由浅到深认识Java语言(6):控制流程语句
该文章Github地址:https://github.com/AntonyCheng/java-notes 在此介绍一下作者开源的SpringBoot项目初始化模板(Github仓库地址:https://github.com/AntonyCheng/spring-boot-init-template & CSDN文章地址:https://blog.c…...
lv17 安防监控项目实战 3
代码目录 框架 our_storage 编译最终生成的目标文件obj 编译生成中间的.o文件 data_global.c 公共资源定义(使用在外extern即可)定义了锁定义了条件变量消息队列id、共享内存id、信号量id及key值发送短信、接收短信的号码向消息队列发送消息的函数&am…...
文本处理基本方法
目录 分词 jieba 词性标注 😆😆😆感谢大家观看😆😆😆 分词 在中文文本中,由于词与词之间没有明显的界限符,如英文中的空格,因此分词是中文自然语言处理的一个基础且…...
Java面试题(Spring篇)
💟💟前言 友友们大家好,我是你们的小王同学😗😗 今天给大家打来的是 Java面试题(Spring篇) 希望能给大家带来有用的知识 觉得小王写的不错的话麻烦动动小手 点赞👍 收藏⭐ 评论📄 小王的主页…...
操作系统:malloc与堆区内存管理
malloc是函数而不是系统调用,他的底层是同调调用brk和mmap这两个系统调用实现功能的,具体选择brk还是mmap要看申请的空间大小以及malloc中的阈值(一般是128kb) 注意申请的空间只有使用才会触发缺页中断映射到物理内存 不理解的话先…...
javaSwing推箱子游戏
一、简介 策略性游戏可以锻炼人的思维能力还能缓解人的压力,使人们暂时忘却生活当中的烦恼,增强人们的逻辑思维能力,游戏的艺术美也吸引着越来越多的玩家和厂商,寓教于乐,在放松人们心情的同时还可以活跃双手。在人类…...
JAVA多线程之JMM
文章目录 1. Java内存模型2. 内存交互3. 三大特性3.1 可见性3.1.1 可见性问题3.1.2 原因3.1.3 解决方法 3.2 原子性3.3 有序性 4. 指令重排5. JMM 与 happens-before5.1 happens-before关系定义5.2 happens-before 关系 在继续学习JUC之前,我们现在这里介绍一下Java…...
Windows10 专业版 系统激活
Windows10 专业版 系统激活 参考: Windows10系统激活技巧 第一步:在电脑桌面,新建一个文本文档 第二步:打开文本文档,输入以下代码后,直接保存关闭文档 slmgr/skms kms.03k.org slmgr/ato 第三步࿱…...
C#使用LINQ和EF Core
在实际应用中,您可以使用 LINQ 查询 EF Core 来执行各种数据库操作。通过 LINQ,您可以轻松地过滤、排序、分组和连接数据。 要使用LINQ查询EF Core中的数据,您可以按照以下步骤进行操作: 首先,确保您已经安装了 Entit…...
数字人解决方案— SadTalker语音驱动图像生成视频原理与源码部署
简介 随着数字人物概念的兴起和生成技术的不断发展,将照片中的人物与音频输入进行同步变得越来越容易。然而,目前仍存在一些问题,比如头部运动不自然、面部表情扭曲以及图片和视频中人物面部的差异等。为了解决这些问题,来自西安…...
HTML5语法总结
文章目录 一.HTML基本框架二.标题标签三.段落标签四.换行与水平线标签五.文本格式化标签(加粗、倾斜、下划线、删除线)六.图像标签扩展:相对路径,绝对路径与在线网址 七.超链接标签八.音频标签九.视频标签十.列表标签十一.表格标签扩展:表格结构标签合并…...
测试微信模版消息推送
进入“开发接口管理”--“公众平台测试账号”,无需申请公众账号、可在测试账号中体验并测试微信公众平台所有高级接口。 获取access_token: 自定义模版消息: 关注测试号:扫二维码关注测试号。 发送模版消息: import requests da…...
从零实现富文本编辑器#5-编辑器选区模型的状态结构表达
先前我们总结了浏览器选区模型的交互策略,并且实现了基本的选区操作,还调研了自绘选区的实现。那么相对的,我们还需要设计编辑器的选区表达,也可以称为模型选区。编辑器中应用变更时的操作范围,就是以模型选区为基准来…...
【第二十一章 SDIO接口(SDIO)】
第二十一章 SDIO接口 目录 第二十一章 SDIO接口(SDIO) 1 SDIO 主要功能 2 SDIO 总线拓扑 3 SDIO 功能描述 3.1 SDIO 适配器 3.2 SDIOAHB 接口 4 卡功能描述 4.1 卡识别模式 4.2 卡复位 4.3 操作电压范围确认 4.4 卡识别过程 4.5 写数据块 4.6 读数据块 4.7 数据流…...
对WWDC 2025 Keynote 内容的预测
借助我们以往对苹果公司发展路径的深入研究经验,以及大语言模型的分析能力,我们系统梳理了多年来苹果 WWDC 主题演讲的规律。在 WWDC 2025 即将揭幕之际,我们让 ChatGPT 对今年的 Keynote 内容进行了一个初步预测,聊作存档。等到明…...
NLP学习路线图(二十三):长短期记忆网络(LSTM)
在自然语言处理(NLP)领域,我们时刻面临着处理序列数据的核心挑战。无论是理解句子的结构、分析文本的情感,还是实现语言的翻译,都需要模型能够捕捉词语之间依时序产生的复杂依赖关系。传统的神经网络结构在处理这种序列依赖时显得力不从心,而循环神经网络(RNN) 曾被视为…...
IoT/HCIP实验-3/LiteOS操作系统内核实验(任务、内存、信号量、CMSIS..)
文章目录 概述HelloWorld 工程C/C配置编译器主配置Makefile脚本烧录器主配置运行结果程序调用栈 任务管理实验实验结果osal 系统适配层osal_task_create 其他实验实验源码内存管理实验互斥锁实验信号量实验 CMISIS接口实验还是得JlINKCMSIS 简介LiteOS->CMSIS任务间消息交互…...
HarmonyOS运动开发:如何用mpchart绘制运动配速图表
##鸿蒙核心技术##运动开发##Sensor Service Kit(传感器服务)# 前言 在运动类应用中,运动数据的可视化是提升用户体验的重要环节。通过直观的图表展示运动过程中的关键数据,如配速、距离、卡路里消耗等,用户可以更清晰…...
【LeetCode】3309. 连接二进制表示可形成的最大数值(递归|回溯|位运算)
LeetCode 3309. 连接二进制表示可形成的最大数值(中等) 题目描述解题思路Java代码 题目描述 题目链接:LeetCode 3309. 连接二进制表示可形成的最大数值(中等) 给你一个长度为 3 的整数数组 nums。 现以某种顺序 连接…...
【前端异常】JavaScript错误处理:分析 Uncaught (in promise) error
在前端开发中,JavaScript 异常是不可避免的。随着现代前端应用越来越多地使用异步操作(如 Promise、async/await 等),开发者常常会遇到 Uncaught (in promise) error 错误。这个错误是由于未正确处理 Promise 的拒绝(r…...
什么是VR全景技术
VR全景技术,全称为虚拟现实全景技术,是通过计算机图像模拟生成三维空间中的虚拟世界,使用户能够在该虚拟世界中进行全方位、无死角的观察和交互的技术。VR全景技术模拟人在真实空间中的视觉体验,结合图文、3D、音视频等多媒体元素…...
