详解Python中的反射机制

Posted mengdie1978

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了详解Python中的反射机制相关的知识,希望对你有一定的参考价值。

详解Python中的反射机制

一、反射的概念
python的反射机制,核心就是利用字符串去已存在的模块中找到指定的属性或方法,找到方法后自动执行——基于字符串的事件驱动。

二、熟悉面向对象的属性方法

  • hasattr(object,\'attrName\'):判断该对象是否有指定名字的属性或方法,返回值是bool类型
  • setattr(object,\'attrName\',value):给指定的对象添加属性以及属性值
  • getattr(object,\'attrName\'):获取对象指定名称的属性或方法,返回值是str类型
  • delattr(object,\'attrName\'):删除对象指定名称的属性或方法值,无返回值
class Fruit:
    # 构造方法
    def __init__(self,name,color):
        self.name = name
        self.color = color
    # 类的普通方法
    def buy(self,price,num):
        print("水果的价格是:",price*num)
"""
    hasattr(object,\'attrName\'):判断该对象是否有指定名字的属性或方法,返回值是bool类型
    setattr(object,\'attrName\',value):给指定的对象添加属性以及属性值
    getattr(object,\'attrName\'):获取对象指定名称的属性或方法,返回值是str类型
    delattr(object,\'attrName\'):删除对象指定名称的属性或方法值,无返回值
"""       
apple = Fruit("苹果","红色")
print(hasattr(apple,\'name\')) # 判断对象是否有该属性或方法
print(hasattr(apple,\'buy\'))

# 获取对象指定的属性值
print(getattr(apple,\'name\'))
print(apple.name)

f = getattr(apple,\'buy\')
f(5,10)
# 设置对象对应的属性
setattr(apple,\'weight\',100)

# 删除对象对应的属性
delattr(apple,\'name\')
print(hasattr(apple,\'name\'))

注:getattr,hasattr,setattr,delattr对模块的修改都在内存中进行,并不会影响文件中真实内容

三、面向对象的反射机制

需求描述:用户通过输入字符串来调用对象的对应方法,通过模拟一个服务器响应用户的请求,设置有注册页、登录页、主页、关于页以及错误页。
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!")

page = WebSite()        
while True:
    choose = input("请输入你要进入的页面>>>")
    if choose == \'register\':
        page.register()
    elif choose == \'login\':
        page.login()
    elif choose == \'home\':
        page.home()
    elif choose == \'about\':
        page.about()
    else:
        page.error()

由于代码段对用户的请求页判断的代码块冗长,并且当新增一个网页时也要实时修改对应的主体代码,维护起来不方便。

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!")

page = WebSite()        
while True:
    choose = input("请输入你要进入的页面>>>")
    # 反射机制实现上述功能,优化代码结构
    if hasattr(page,choose):
        f = getattr(page,choose)
    else:
        page.error()

通过应用面向对象操作属性的方法很好的解决了这个问题,这就是反射机制。Python面向对象中的反射:通过字符串的形式操作对象相关的属性。

四、实例应用

在正式介绍实际web开发中的应用场景之前,先来看看这样的一个模块。

# 模块:importlib
import importlib
res = \'myfile.b\'
# 传入字符串路径,调用importlib类,实现 from myfile import b
ret = importlib.import_module(res) 
# 该方法最小只能到.py文件名即模块
print(ret)

是的,稍微阅读一下上面的实例代码就可以知道importlib模块的功能是:即安装字符串式的路径自动导入模块,调用importlib.import_module()方法,但该方法的最小粒度只能达到.py文件名即模块。

需求描述:输入多层的模块路径,自动生成对象并调用该类的方法。比如:notify.email.Email,notify包下面有模块email,模块email中包括了Email类,利用该类声明对象,并调用其中的send()方法。
import importlib
#\'notify.email.Email\'
path_str = input("请输入包-模块-类的字符串路径:")
module_path,class_name = path_str.rsplit(\'.\',maxsplit=1)
# 1 利用字符串导入模块
module = importlib.import_module(module_path)  # from notify import email
# 2 利用反射获取类名
cls = getattr(module,class_name)  # Email、QQ、Wechat
# 3 生成类的对象
obj = cls()
# 4 直接调用send方法
obj.send()

动态加载模块,python反射机制

参考技术A

动态加载模块用的模块importlib 里面的import_module(“字符串模块路径”, 相对路径包名)

2.getattr(对象,“字符串属性”,默认值)

补充

任意创建一个文件夹,在里面同级创建f1-f4.py的四个文件。

内容如下:

f1.py

f2.py

f3.py

f4.py

f5.py

f4.py:通过动态导入模块,动态添加属性 删除。可以解决接口自动化中的动态传参

F5.py: 通过传参的方式,去获取类的实例方法,可以做到不知道需要哪个方法的时候,用字符串的方法去调用需要的方法。接口自动化中参数化的时候可以用到。

反射机制,当通过类名去反射得到的实例方法,需要传类名进去。通过self或者对象得到的实例方法,不需要传self或者类名。如果是实例对象或者类 去调用静态方法, 都不用给函数传self cls

以上是关于详解Python中的反射机制的主要内容,如果未能解决你的问题,请参考以下文章

动态加载模块,python反射机制

Java反射机制详解

Java反射机制详解

Python的反射机制hasattr() getattr() setattr() 函数使用方法详解

[转]Java反射机制详解

Java 反射详解 转载