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

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)。想象一下,你正在编写一段代码,并希望这段代码能够具备自我认知和动态调整的能力。就好比一面镜子,能反映出它自身的属性和行为。在编程领域&#xff0…...

Python数学建模-2.9Matplotlib库

Matplotlib库是Python中一个非常流行的绘图库,它提供了大量的绘图工具,可以生成各种类型的静态、动态、交互式的图表。Matplotlib的设计初衷是为了与NumPy配合使用,从而提供一个强大的数学绘图工具。 1.Matplotlib的主要特点 丰富的图表类型…...

【MySQL】数据库的基础概念

👦个人主页:Weraphael ✍🏻作者简介:目前学习计网、mysql和算法 ✈️专栏:MySQL学习 🐋 希望大家多多支持,咱一起进步!😁 如果文章对你有帮助的话 欢迎 评论&#x1f4ac…...

Linux:离线安装 jdk-8(配置Java环境)

Linux:离线安装 jdk-8(配置Java环境) 1、jdk简介2、检查已安装的Java版本,并卸载3、准备安装包4、解压安装包、进行安装5、设置环境变量(全局/个人) 💖The Begin💖点点关注,收藏不迷路&#x1f…...

【DP】第十三届蓝桥杯省赛C++ B组《李白打酒加强版》(C++)

【题目描述】 话说大诗人李白,一生好饮。 幸好他从不开车。 一天,他提着酒壶,从家里出来,酒壶中有酒 2 斗。 他边走边唱: 无事街上走,提壶去打酒。 逢店加一倍,遇花喝一斗。 这一路上&am…...

数据结构试卷第九套

1.时间复杂度 2.树,森林,二叉树的转换 2.1树转二叉树 给所有的兄弟节点之间加一条连线;去线,只保留当前根节点与第一个叶子节点的连线,删除它与其他节点之间的连线;然后根据左孩子右兄弟进行调整&#xf…...

【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 配置文件&#xff0…...

由浅到深认识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 第三步&#xff1…...

C#使用LINQ和EF Core

在实际应用中,您可以使用 LINQ 查询 EF Core 来执行各种数据库操作。通过 LINQ,您可以轻松地过滤、排序、分组和连接数据。 要使用LINQ查询EF Core中的数据,您可以按照以下步骤进行操作: 首先,确保您已经安装了 Entit…...

数字人解决方案— SadTalker语音驱动图像生成视频原理与源码部署

简介 随着数字人物概念的兴起和生成技术的不断发展,将照片中的人物与音频输入进行同步变得越来越容易。然而,目前仍存在一些问题,比如头部运动不自然、面部表情扭曲以及图片和视频中人物面部的差异等。为了解决这些问题,来自西安…...

HTML5语法总结

文章目录 一.HTML基本框架二.标题标签三.段落标签四.换行与水平线标签五.文本格式化标签(加粗、倾斜、下划线、删除线)六.图像标签扩展:相对路径,绝对路径与在线网址 七.超链接标签八.音频标签九.视频标签十.列表标签十一.表格标签扩展:表格结构标签合并…...

UE5 学习系列(二)用户操作界面及介绍

这篇博客是 UE5 学习系列博客的第二篇,在第一篇的基础上展开这篇内容。博客参考的 B 站视频资料和第一篇的链接如下: 【Note】:如果你已经完成安装等操作,可以只执行第一篇博客中 2. 新建一个空白游戏项目 章节操作,重…...

WordPress插件:AI多语言写作与智能配图、免费AI模型、SEO文章生成

厌倦手动写WordPress文章?AI自动生成,效率提升10倍! 支持多语言、自动配图、定时发布,让内容创作更轻松! AI内容生成 → 不想每天写文章?AI一键生成高质量内容!多语言支持 → 跨境电商必备&am…...

在鸿蒙HarmonyOS 5中使用DevEco Studio实现录音机应用

1. 项目配置与权限设置 1.1 配置module.json5 {"module": {"requestPermissions": [{"name": "ohos.permission.MICROPHONE","reason": "录音需要麦克风权限"},{"name": "ohos.permission.WRITE…...

mysql已经安装,但是通过rpm -q 没有找mysql相关的已安装包

文章目录 现象:mysql已经安装,但是通过rpm -q 没有找mysql相关的已安装包遇到 rpm 命令找不到已经安装的 MySQL 包时,可能是因为以下几个原因:1.MySQL 不是通过 RPM 包安装的2.RPM 数据库损坏3.使用了不同的包名或路径4.使用其他包…...

uniapp 字符包含的相关方法

在uniapp中,如果你想检查一个字符串是否包含另一个子字符串,你可以使用JavaScript中的includes()方法或者indexOf()方法。这两种方法都可以达到目的,但它们在处理方式和返回值上有所不同。 使用includes()方法 includes()方法用于判断一个字…...

解决:Android studio 编译后报错\app\src\main\cpp\CMakeLists.txt‘ to exist

现象: android studio报错: [CXX1409] D:\GitLab\xxxxx\app.cxx\Debug\3f3w4y1i\arm64-v8a\android_gradle_build.json : expected buildFiles file ‘D:\GitLab\xxxxx\app\src\main\cpp\CMakeLists.txt’ to exist 解决: 不要动CMakeLists.…...

MySQL 索引底层结构揭秘:B-Tree 与 B+Tree 的区别与应用

文章目录 一、背景知识:什么是 B-Tree 和 BTree? B-Tree(平衡多路查找树) BTree(B-Tree 的变种) 二、结构对比:一张图看懂 三、为什么 MySQL InnoDB 选择 BTree? 1. 范围查询更快 2…...

ubuntu22.04 安装docker 和docker-compose

首先你要确保没有docker环境或者使用命令删掉docker sudo apt-get remove docker docker-engine docker.io containerd runc安装docker 更新软件环境 sudo apt update sudo apt upgrade下载docker依赖和GPG 密钥 # 依赖 apt-get install ca-certificates curl gnupg lsb-rel…...

RLHF vs RLVR:对齐学习中的两种强化方式详解

在语言模型对齐(alignment)中,强化学习(RL)是一种重要的策略。而其中两种典型形式——RLHF(Reinforcement Learning with Human Feedback) 与 RLVR(Reinforcement Learning with Ver…...

Shell 解释器​​ bash 和 dash 区别

bash 和 dash 都是 Unix/Linux 系统中的 ​​Shell 解释器​​,但它们在功能、语法和性能上有显著区别。以下是它们的详细对比: ​​1. 基本区别​​ ​​特性​​​​bash (Bourne-Again SHell)​​​​dash (Debian Almquist SHell)​​​​来源​​G…...