python--反射的使用--getattr,hasattr【,setattr,delattr】

本文介绍了Python反射的使用,它能让调用更简单。主要内容包括获取类中的属性和方法、调用导入模块的属性和方法,以及反射自己模块的属性和方法。还提到了hasattr用于判断,其参数需与getattr一致。

反射的使用使调用更加简单

# 反射.py
class A:
	def __init__(self, name, age, gender):
		self.name = name
		self.age = age
		self.gender = gender
	def work(self):
		return self.name+'的工作是吃饭'
a = A('张三', 12, '男')

加入现在有一个类如上,并实例化了
现在需要调用a的姓名的话,通常的做法就是

a.name

使用反射

getattr(a, 'name')

看着好像更麻烦了,但是并不是那么简单的
比如我输入name,给我返回name的值,输入age,返回age的值。。。。

ret = input('>>> ')
print(getattr(a, ret))

结果如下:
在这里插入图片描述
一、获取类中的属性和方法

# 属性
getattr(a, 'name')  # 张三
# 方法
ret = getattr(a, 'work')  # <bound method A.work of <__main__.A object at 0x02E7E670>>
ret()  # 调用   ---> 张三的工作是吃饭

二、调用导入模块的属性和方法
假如有一个mygetattrtext.py文件

day = 2018

class Person:
	name = '李四'
	age = 30
	gender = '男'

	def work(self):
		return '看电视'
# 反射.py
from mygetattrtest import Person, day  # 导入模块

p = Person()  # 实例化对象

# 属性
getattr(p, 'name')  # 李四
# 方法
ret = getattr(p, 'work')
ret()  # 看电视

# ----------------
# 反射导入模块的day属性
import sys
from mygetattrtest import day
getattr(sys.modules[__name__], 'day')

三、反射自己的模块的属性和方法

# 反射.py
year = 2019

def eat():
	return '我正在吃饭'

import sys

# 反射自己模块的属性
getattr(sys.modules[__name__], 'year')  # 2019

# 反射自己模块的方法(函数)
ret = getattr(sys.modules[__name__], 'eat')
ret()  # 我正在吃饭

hasattr
判断有没有呗

class A:
	def __init__(self, name, age, gender):
		self.name = name
		self.age = age
		self.gender = gender
	def work(self):
		return self.name+'的工作是吃饭'

# print(A.name)  # 一般的写法

a = A('张三', 12, '男')

# --------------------------------hasattr

if hasattr(a, 'name'):
	print(getattr(a, 'name'))  # 张三
else:
	print('找不到')

注意:hasattr中的参数和getattr中的参数是一致的,否则判断不就没用了吗

setattr和selattr不说了

<think>我们被要求介绍Python中的hasattrgetattr方法,包括它们的介绍和使用方法。根据引用资料,我们可以总结如下: 1. hasattr(obj, name): 用于检查对象obj是否具有名为name的属性或方法,返回布尔值。 2. getattr(obj, name[, default]): 用于获取对象obj中名为name的属性或方法。如果存在,返回其值;如果不存在且提供了default参数,则返回default;否则抛出AttributeError。 另外,引用中还提到了setattrdelattr,但用户本次只询问了hasattrgetattr使用场景: - 动态地访问对象的属性或方法,特别是在编写通用代码或处理未知对象结构时。 - 避免在访问属性前进行冗余的条件判断。 注意事项: - 对于方法(函数属性),getattr返回的是函数对象本身,需要调用才能执行。 - 如果属性不存在且没有提供默认值,会引发异常,因此需要谨慎处理。 下面我们通过示例来演示这两个函数的使用。 </think>### Python中`hasattr`与`getattr`方法详解 `hasattr()`和`getattr()`是Python内置的属性操作函数,主要用于**动态访问对象属性**,在元编程、反射机制中应用广泛。下面从核心概念到具体应用进行系统讲解: --- #### 1. 核心功能对比 | **函数** | **功能描述** | **返回值** | **异常处理** | |----------------|----------------------------------------------------------------------------|--------------------|--------------------------------| | `hasattr(obj, name)` | 检查对象`obj`是否包含名为`name`的属性 | `True`/`False` | 永不抛出异常 | | `getattr(obj, name[, default])` | 获取对象`obj`中名为`name`的属性值 | 属性值或默认值 | 无默认值时抛出`AttributeError` | 数学描述属性存在性验证: 设对象 $O$,属性名 $A$,则存在性验证为: $$ \text{hasattr}(O, A) = \begin{cases} \text{True} & \text{if } A \in \text{attributes}(O) \\ \text{False} & \text{otherwise} \end{cases} $$ --- #### 2. 使用场景与示例 ##### ▶ 场景1:安全访问未知属性 ```python class User: def __init__(self, name): self.name = name user = User("Alice") # 安全检查 + 获取属性 if hasattr(user, "email"): # 检查属性是否存在[^1] print(getattr(user, "email")) # 若存在则获取 else: print("No email attribute") # 输出: No email attribute # 带默认值的获取 age = getattr(user, "age", 25) # 属性不存在返回25[^5] print(age) # 输出: 25 ``` ##### ▶ 场景2:动态调用方法 ```python class Calculator: def add(self, a, b): return a + b def multiply(self, a, b): return a * b calc = Calculator() method_name = "multiply" # 方法名可动态配置 if hasattr(calc, method_name): func = getattr(calc, method_name) # 获取方法对象[^4] result = func(3, 4) # 执行方法 print(f"3 * 4 = {result}") # 输出: 3 * 4 = 12 ``` ##### ▶ 场景3:插件系统集成 ```python # 动态加载并执行插件方法 plugins = [{"name": "Logger", "method": "log_data"}] for plugin in plugins: plugin_obj = __import__(plugin["name"]) # 动态导入模块 if hasattr(plugin_obj, plugin["method"]): # 验证方法存在[^2] getattr(plugin_obj, plugin["method"])("Operation completed") # 执行方法 ``` --- #### 3. 关键注意事项 1. **方法调用差异** - 获取实例方法需通过实例对象:`getattr(obj_instance, 'method')()` - 获取类方法需通过类对象:`getattr(ClassName, 'class_method')()`[^5] 2. **性能优化** 在循环中避免重复调用`hasattr()`,优先缓存结果: ```python # 优化前(每次循环执行检查) for item in large_list: if hasattr(item, "process"): getattr(item, "process")() # 优化后(缓存检查结果) processable = [item for item in large_list if hasattr(item, "process")] for item in processable: getattr(item, "process")() ``` 3. **继承链查找** 属性查找遵循方法解析顺序(MRO),包含继承属性: ```python class Parent: shared = "Parent Value" class Child(Parent): pass child = Child() print(hasattr(child, "shared")) # 输出: True[^3] print(getattr(child, "shared")) # 输出: Parent Value ``` --- #### 4. 典型应用场景 - **动态配置处理**:根据配置文件动态调用不同类的方法 - **API版本兼容**:检查新版对象是否支持旧版属性 - **插件架构**:运行时加载并验证扩展模块功能 - **数据序列化**:通用JSON转换器动态获取对象字段 - **单元测试**:验证Mock对象是否包含特定方法 > 注意:`getattr()`和`hasattr()`仅访问**直接绑定属性**,不触发描述符协议或`__getattr__`魔法方法。如需完整属性访问控制,应配合`__getattribute__`使用[^2][^5]。 --- ###
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

秒不可闫M先生

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值