Python基础学习之反射机制详解
作者:TtrOps 发布时间:2023-02-15 11:02:01
什么是反射
在Python中,反射是指通过一组内置的函数和语句,在运行时动态地访问、检查和修改对象的属性、方法和类信息的机制。Python中的反射机制非常强大,可以使程序更加灵活和可扩展。
Python中的反射主要涉及以下几个内置函数和语句:
getattr():获取对象的属性或方法。可以通过对象和字符串的方式传递属性或方法名,并且还可以提供一个默认值,用于在属性或方法不存在时返回。
setattr():设置对象的属性或方法。可以通过对象、字符串和值的方式传递属性或方法名和值。
delattr():删除对象的属性或方法。可以通过对象和字符串的方式传递属性或方法名。
dir():获取对象的所有属性和方法的列表。可以使用dir()函数来获取对象的所有属性和方法的列表。
type():获取对象的类型。可以使用type()函数来获取对象的类型信息。
inspect模块:该模块提供了更加高级的反射功能,可以用于获取函数和类的参数列表、注解、源代码等信息。
应用场景
反射在Python中的应用场景非常广泛,例如:
动态加载模块和类:使用反射可以在运行时动态加载模块和类,以便于程序更加灵活和可扩展。
动态修改对象属性和方法:使用反射可以在运行时动态修改对象的属性和方法,以便于程序更加灵活。
实现插件系统:使用反射可以实现插件系统,允许程序在运行时动态加载和卸载插件。
实现ORM框架:使用反射可以实现ORM框架,允许程序在运行时动态地将Python对象映射到数据库中的表格。
总之,反射是Python中一种非常有用的元编程技术,可以使程序更加灵活和可扩展。但是,在使用反射时需要谨慎,避免滥用,因为反射可能会影响性能并增加代码复杂度。
基本小栗子
1.访问对象属性
class MyClass:
def __init__(self, x):
self.x = x
obj = MyClass(42)
attr_name = "x"
attr_value = getattr(obj, attr_name)
print(f"{attr_name} = {attr_value}")
2.动态调用对象方法
class MyClass:
def my_method(self, x, y):
return x + y
my_object = MyClass()
result = getattr(my_object, "my_method")(1, 2)
print(result) # 输出 3
3.动态创建对象
class MyClass:
def __init__(self, x, y):
self.x = x
self.y = y
my_class = type("MyClass", (), {"x": 1, "y": 2})
my_object = my_class()
print(my_object.x, my_object.y) # 输出 1 2
4.动态导入模块
# 使用 importlib.import_module() 导入模块
import importlib
module_name = 'math'
module = importlib.import_module(module_name)
# 使用 getattr() 访问模块的属性
pi_value = getattr(module, 'pi')
print(pi_value) # 输出: 3.141592653589793
5.获取类属性
class MyClass:
my_class_attribute = "Hello World"
print(getattr(MyClass, "my_class_attribute")) # 输出 "Hello World"
6.检查对象是否具有属性
class MyClass:
def __init__(self):
self.my_attribute = "Hello World"
my_object = MyClass()
print(hasattr(my_object, "my_attribute")) # 输出 True
print(hasattr(my_object, "non_existent_attribute")) # 输出 False
7.动态获取类的方法列表
class MyClass:
def __init__(self):
self.my_attribute = 'Hello, World!'
def my_method(self):
print(self.my_attribute)
# 使用 dir() 获取类的方法列表
method_list = [method_name for method_name in dir(MyClass) if callable(getattr(MyClass, method_name))]
print(method_list) # 输出: ['__init__', '__module__', 'my_method']
8.动态调用模块中的函数
# 使用 importlib.import_module() 导入模块
import importlib
module_name = 'math'
module = importlib.import_module(module_name)
# 使用 getattr() 访问模块中的函数
sqrt_function = getattr(module, 'sqrt')
result = sqrt_function(4)
print(result) # 输出: 2.0
9.动态修改对象的属性
class MyClass:
def __init__(self):
self.my_attribute = 'Hello, World!'
my_object = MyClass()
# 使用 setattr() 修改对象的属性
setattr(my_object, 'my_attribute', 'Hello, Universe!')
print(my_object.my_attribute) # 输出: 'Hello, Universe!'
贴近实际应用的小场景
假设正在构建一个电商网站,并需要实现一个订单管理系统。这个系统需要支持多种订单类型(例如普通订单、抢购订单、团购订单等),每种订单类型有其独特的属性和方法。
为了实现这个系统,可以使用反射来动态地创建订单对象,并根据订单类型来调用相应的属性和方法。
首先,需要定义一个基本的订单类,该类包含所有订单类型的通用属性和方法。然后,可以创建一个名为 OrderFactory 的工厂类,该类负责根据订单类型创建订单对象。
下面是示例代码:
class Order:
def __init__(self, order_id, customer_name, product_id):
self.order_id = order_id
self.customer_name = customer_name
self.product_id = product_id
def calculate_total_price(self):
# 计算订单总价
pass
def validate_order(self):
# 验证订单是否合法
pass
def confirm_order(self):
# 确认订单
pass
class OrderFactory:
@staticmethod
def create_order(order_type, order_id, customer_name, product_id):
# 动态创建订单对象
order_class = globals().get(order_type)
if not order_class:
raise ValueError(f"Invalid order type: {order_type}")
return order_class(order_id, customer_name, product_id)
class FlashSaleOrder(Order):
def __init__(self, order_id, customer_name, product_id, discount):
super().__init__(order_id, customer_name, product_id)
self.discount = discount
def calculate_total_price(self):
# 计算限时抢购订单的总价(包含折扣)
pass
class GroupBuyOrder(Order):
def __init__(self, order_id, customer_name, product_id, group_size):
super().__init__(order_id, customer_name, product_id)
self.group_size = group_size
def calculate_total_price(self):
# 计算团购订单的总价(根据购买人数和商品单价)
pass
现在,可以使用 OrderFactory 来创建订单对象。例如,要创建一个限时抢购订单,可以使用以下代码:
order_type = "FlashSaleOrder"
order_id = "123"
customer_name = "Alice"
product_id = "456"
discount = 0.2
order = OrderFactory.create_order(order_type, order_id, customer_name, product_id, discount)
这将动态地创建一个 FlashSaleOrder 对象,并使用提供的参数初始化它。
另外,如果需要动态调用订单对象的方法,可以使用 Python 的内置反射机制。例如,要调用订单对象的 calculate_total_price 方法,可以使用以下代码:
method_name = "calculate_total_price"
method = getattr(order, method_name)
total_price = method()
这将动态地获取 order 对象的 calculate_total_price 方法,并调用它来计算订单的总价。
来源:https://mp.weixin.qq.com/s/NRSkzdAbcdF828YhFd88NA
猜你喜欢
- 今天我们来介绍下Python基础教程学习之iter() 方法另外的用法。据说很少有人知道这个用法!一、上代码、学用法我们都比较熟悉 iter
- 当管理SQL Server内在的帐户和密码时,我们很容易认为这一切都相当的安全。毕竟,你的SQL Server系统被保护在防火墙里,而且还有
- <?php /* *文件名:linearList.php * 功能:数据结构线性表的顺序存储实现 * author:黎锦焕 * @co
- 本文实例讲述了python登录豆瓣并发帖的方法。分享给大家供大家参考。具体如下:这里涉及urllib、urllib2及cookielib常用
- 你可能正建立自己在网上的家-WEB页,又或者它已经被你安置在网络某一角里。无论出于什么目的,你都希望有
- 本文实例为大家分享了python实现事件驱动的具体代码,供大家参考,具体内容如下EventManager事件管理类实现,大概就百来行代码左右
- 今天有朋友问到如下一则案例,ORA-01114,ORA-27067以及OSD-04026错误同时出现:*** ACTION NAME:()
- 下面发一个简单的在线调试服务端js代码的asp源码。并可以提示代码具体错误信息。<%@language="javascrip
- 微软昨天在其2009年专业开发者大会上展示了下一个版本的Internet Explorer浏览器IE9。尽管只是一个早期版本,IE开发团队还
- write()方法把字符串str写入文件。没有返回值。由于缓冲,字符串可能不实际显示文件,直到flush()或close()方法
- 可迭代(iterable)迭代(遍历)就是按照某种顺序逐个访问对象中的每一项。Python中有很多对象都是可以通过for语句来直接遍历的,例
- 我的操作系统为centos6.51 首先选择django要使用什么数据库。django1.10默认数据库为sqlite3,本人想
- 如下所示:两个函数:Basemap.drawparallels ##纬度 Basemap.drawmeridia
- 安装npm install --save reduxnpm install --save redux-saga配置actionactionT
- 本文实例总结了Python实现string字符串连接的方法。分享给大家供大家参考,具体如下:以下基于python 2.7版本,代码片段真实有
- 一、为什么要安装虚拟环境 情景一、项目A需要某个库的1.0版本,项目B需要这个库的2.0版本。如果没有安装虚拟环境
- 当你使用UPDATE, INSERT, DELETE语句更新数据的时候,你就改变了两个地方的数据:log buffer和data buffe
- 安装pip install faker使用简单使用本库可生成姓名、地址、电话、邮箱、公司等等一系列数据。首先导入库,实例化:from fak
- 代码如下:--Begin Index(索引) 分析优化的相关 Sql -- 返回当前数据库所有碎片率大于25%的索引 -- 运行
- 经常看到说正则的文章,但说的只是方法,却很少有说以下几个基本概念:1.贪婪:+,*,?,{m,n}等默认是贪婪匹配,即尽可能多匹配,也叫最大