Python动态属性教程_setattr与getattr用法

12次阅读

python中动态设置和获取属性核心靠setattr()和getattr():前者按字符串名设属性(支持新增),后者按字符串名取值并可设默认值,二者配合__setattr__和__getattr__可实现属性访问的精细控制。

Python动态属性教程_setattr与getattr用法

Python中动态设置和获取属性,核心靠setattr()getattr()两个内置函数。它们让对象属性的操作不再局限于写死的点号访问,而是能用字符串控制,特别适合配置驱动、序列化、ORM映射或插件系统等场景。

setattr:按名字给对象设属性

setattr(obj, name, value) 等价于 obj.name = value,但name是字符串,可动态生成。

  • 第一个参数必须是**已存在的对象**(不能是类名本身,除非你想改类属性)
  • 第二个参数是**字符串形式的属性名**,支持不存在的属性——会直接新增
  • 第三个参数是任意值,包括函数、Lambda、甚至另一个对象

示例:

class Person:     pass 

p = Person() setattr(p, 'name', 'Alice') setattr(p, 'age', 30) setattr(p, 'say_hello', lambda: print('Hi!'))

print(p.name) # Alice print(p.age) # 30 p.say_hello() # Hi!

getattr:安全地按名字取属性值

getattr(obj, name[, default]) 类似 obj.name,但更灵活:支持默认值,且不会因属性不存在而报错。

立即学习Python免费学习笔记(深入)”;

  • 如果name对应属性存在,返回其值
  • 如果不存在,且提供了default,就返回它(比如None、空列表、0等)
  • 如果没提供default且属性不存在,则抛出AttributeError

常见用法:

# 安全读取配置字段 config = {'host': 'localhost', 'port': 8000} host = getattr(config, 'host', '127.0.0.1')   # 注意:config是dict,不支持getattr → 这里应是自定义对象 # 正确做法:先定义类或使用types.SimpleNamespace from types import SimpleNamespace cfg = SimpleNamespace(host='localhost', port=8000) host = getattr(cfg, 'host', '127.0.0.1')      # 'localhost' timeout = getattr(cfg, 'timeout', 30)        # 30(默认值)

配合__setattr__和__getattr__实现更精细控制

仅靠setattr/getattr是外部操作;若想在属性被设或取时自动触发逻辑(如日志、类型检查、懒加载),需重写特殊方法:

  • __setattr__(self, name, value):每次self.x = ysetattr(self, ...)都会调用它
  • __getattr__(self, name):仅当属性**确实不存在**时才被调用(注意不是__getattribute__

小提醒:重写__setattr__时,记得用super().__setattr__(...)赋值,否则可能无限递归

class LoggedPerson:     def __setattr__(self, name, value):         print(f'Setting {name} = {value}')         super().__setattr__(name, value)  # 关键:用super避免递归 

p = LoggedPerson() p.name = 'Bob' # 输出:Setting name = Bob

实际应用场景举例

这些函数不是炫技,而是解决真实问题的工具

  • 从字典批量初始化对象for k, v in data.items(): setattr(obj, k, v)
  • 实现简易ORM字段映射:把数据库列名转为对象属性,用getattr(row, 'user_id')取值
  • 命令行参数绑定到对象:argparse结果用setattr(cfg, opt.dest, opt.value)存入配置对象
  • 插件式功能注册:根据字符串名动态挂载处理函数:setattr(self, 'on_save', handler_func)

不复杂但容易忽略。用对了,代码立刻变灵活;乱用则容易引发隐晦bug,比如误覆写内置属性或拼错字符串名。

text=ZqhQzanResources