published on in tech
tags: python

Reflection in Python

Python Reflection - Python中的反射

我们常常会遇到这样的需求:需要执行对象里的某个方法,或需要调用对象中的某个变量,但是由于种种原因我们无法确定这个方法或变量是否存在,这是我们需要用一个特殊的方法或机制要访问和操作这个未知的方法或变量

常见场景

  1. 动态加载模块
  2. Web框架中的URL路由
  3. 你的伙伴封装好了很多py文件,然后你负责根据不同需求去调用不同的模块
  4. 你写了一个类,其中封装了很多方法,这是你需要提供一个统一的入口供他人调用(类似路由转发)

Python中的反射

1. __import__() #动态加载模块
2. hasattr() #判断实例是否存在字符串对应的属性
3. gettattr #获取属性
4. setattr() #将对应字符串的已存在的属性添加到实例当中
5. delattr() #删除属性

Sample 1

# 动态导入func_001模块
func001_module = __import__('src.func_001', 'fromlist=True)
# 判断是否存在对应类
if hasattr(func001_module, 'Func001'):
    # 创建Func001类实例对象
    func001 = getattr(func001_module, 'Func001')
    # 判断是否存在process成员方法
    if hasattr(func001, 'process'):
        # 获取process方法
        func001_process = getattr(func001, 'process')
        # 执行
        func001_process(func001)

Sample 2

一个类中:

def add(self):
    pass
def delete(self):
    pass
def update(self):
    pass
...

def before_run(self, method):
    if hasattr(self, method):
        func = getattr(self, method)
        func()

为什么不建议使用eval()

反射就是通过字符串的形式,导入模块;通过字符串的形式,去模块寻找指定函数,并执行。利用字符串的形式去对象(模块)中操作(查找/获取/删除/添加)成员,一种基于字符串的事件驱动!

有时候也是为了防止使用eval这种比较危险的函数。eval由于直接执行的就是命令,是有可能带来安全风险的。