当前位置: 首页 > 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基本框架二.标题标签三.段落标签四.换行与水平线标签五.文本格式化标签(加粗、倾斜、下划线、删除线)六.图像标签扩展:相对路径,绝对路径与在线网址 七.超链接标签八.音频标签九.视频标签十.列表标签十一.表格标签扩展:表格结构标签合并…...

测试微信模版消息推送

进入“开发接口管理”--“公众平台测试账号”,无需申请公众账号、可在测试账号中体验并测试微信公众平台所有高级接口。 获取access_token: 自定义模版消息: 关注测试号:扫二维码关注测试号。 发送模版消息: import requests da…...

SCAU期末笔记 - 数据分析与数据挖掘题库解析

这门怎么题库答案不全啊日 来简单学一下子来 一、选择题(可多选) 将原始数据进行集成、变换、维度规约、数值规约是在以下哪个步骤的任务?(C) A. 频繁模式挖掘 B.分类和预测 C.数据预处理 D.数据流挖掘 A. 频繁模式挖掘:专注于发现数据中…...

在WSL2的Ubuntu镜像中安装Docker

Docker官网链接: https://docs.docker.com/engine/install/ubuntu/ 1、运行以下命令卸载所有冲突的软件包: for pkg in docker.io docker-doc docker-compose docker-compose-v2 podman-docker containerd runc; do sudo apt-get remove $pkg; done2、设置Docker…...

rnn判断string中第一次出现a的下标

# coding:utf8 import torch import torch.nn as nn import numpy as np import random import json""" 基于pytorch的网络编写 实现一个RNN网络完成多分类任务 判断字符 a 第一次出现在字符串中的位置 """class TorchModel(nn.Module):def __in…...

Hive 存储格式深度解析:从 TextFile 到 ORC,如何选对数据存储方案?

在大数据处理领域,Hive 作为 Hadoop 生态中重要的数据仓库工具,其存储格式的选择直接影响数据存储成本、查询效率和计算资源消耗。面对 TextFile、SequenceFile、Parquet、RCFile、ORC 等多种存储格式,很多开发者常常陷入选择困境。本文将从底…...

Selenium常用函数介绍

目录 一,元素定位 1.1 cssSeector 1.2 xpath 二,操作测试对象 三,窗口 3.1 案例 3.2 窗口切换 3.3 窗口大小 3.4 屏幕截图 3.5 关闭窗口 四,弹窗 五,等待 六,导航 七,文件上传 …...

【JavaSE】多线程基础学习笔记

多线程基础 -线程相关概念 程序(Program) 是为完成特定任务、用某种语言编写的一组指令的集合简单的说:就是我们写的代码 进程 进程是指运行中的程序,比如我们使用QQ,就启动了一个进程,操作系统就会为该进程分配内存…...

NPOI操作EXCEL文件 ——CAD C# 二次开发

缺点:dll.版本容易加载错误。CAD加载插件时,没有加载所有类库。插件运行过程中用到某个类库,会从CAD的安装目录找,找不到就报错了。 【方案2】让CAD在加载过程中把类库加载到内存 【方案3】是发现缺少了哪个库,就用插件程序加载进…...

django blank 与 null的区别

1.blank blank控制表单验证时是否允许字段为空 2.null null控制数据库层面是否为空 但是,要注意以下几点: Django的表单验证与null无关:null参数控制的是数据库层面字段是否可以为NULL,而blank参数控制的是Django表单验证时字…...

给网站添加live2d看板娘

给网站添加live2d看板娘 参考文献: stevenjoezhang/live2d-widget: 把萌萌哒的看板娘抱回家 (ノ≧∇≦)ノ | Live2D widget for web platformEikanya/Live2d-model: Live2d model collectionzenghongtu/live2d-model-assets 前言 网站环境如下,文章也主…...