python反射

一、反射

反射,提供了一种更加灵活的方式让你可以实现去 对象 中操作成员(以字符串的形式去 对象 中进行成员的操作)。

正常面向对象使用,如下创建一个类

复制代码
class Person(object):

    def __init__(self, name, wx):
        self.name = name
        self.wx = wx

    def show(self):
        message = "姓名{},微信:{}".format(self.name, self.wx)

然后创建一个对象,通过对象调用成员

复制代码
    user_object = Person("张三", "12345678")

    # 对象.成员 的格式去获取数据
    var = user_object.name
    var = user_object.wx
    user_object.show()

    # 对象.成员 的格式无设置数据
    user_object.name = "李四"

反射使用

复制代码
    user = Person("张三", "12345678")

    # getattr 获取成员
    getattr(user, "name")  # user.name
    getattr(user, "wx")  # user.wx

    method = getattr(user, "show")  # user.show
    method()
    # 或
    getattr(user, "show")()

    # setattr 设置成员
    setattr(user, "name", "李四")  # user.name = "吴培期"
    print(user.name)  # 李四

Python中提供了4个内置函数来支持反射:

  • getattr,去对象中获取成员

    复制代码
    v1 = getattr(对象,"成员名称")
    v2 = getattr(对象,"成员名称", 不存在时的默认值)
  • setattr,去对象中设置成员

    复制代码
    setattr(对象,"成员名称",值)
  • hasattr,对象中是否包含成员

    复制代码
    v1 = hasattr(对象,"成员名称") # True/False
  • delattr,删除对象中的成员

    复制代码
    delattr(对象,"成员名称")

以后如果再遇到 对象.成员 这种编写方式时,均可以基于反射来实现。

案例

python 复制代码
class Account(object):

    def login(self):
        pass

    def register(self):
        pass

    def index(self):
        pass

    
def run(self):
    name = input("请输入要执行的方法名称:") # index register login xx run ..
    
    account_object = Account()
    method = getattr(account_object, name,None) # index = getattr(account_object,"index")
    
    if not method:
        print("输入错误")
        return 
    method()

二、一些皆对象

在Python中有这么句话:一切皆对象。 每个对象的内部都有自己维护的成员。

  • 对象是对象

    python 复制代码
    class Person(object):
        
        def __init__(self,name,wx):
            self.name = name
            self.wx = wx
    	
        def show(self):
            message = "姓名{},微信:{}".format(self.name,self.wx)
            
            
    user_object = Person("张三","123456")
    user_object.name
  • 类是对象

    python 复制代码
    class Person(object):
        title = "张三"
    
    Person.title
    # Person类也是一个对象(平时不这么称呼)
  • 模块是对象

    python 复制代码
    import re
    
    re.match
    # re模块也是一个对象(平时不这么称呼)。

由于反射支持以字符串的形式去对象中操作成员【等价于 对象.成员 】,所以,基于反射也可以对类、模块中的成员进行操作。

简单粗暴:只要看到 xx.oo 都可以用反射实现。

python 复制代码
class Person(object):
    title = "张三"

v1 = Person.title
print(v1)
v2 = getattr(Person,"title")
print(v2)
python 复制代码
import re

v1 = re.match("\w+","dfjksdufjksd")
print(v1)

func = getattr(re,"match")
v2 = func("\w+","dfjksdufjksd")
print(v2)

三、import_module + 反射

在Python中如果想要导入一个模块,可以通过import语法导入;也可以通过字符串的形式导入。

示例一:

python 复制代码
# 导入模块
import random

v1 = random.randint(1,100)
python 复制代码
# 导入模块
from importlib import import_module

m = import_module("random")

v1 = m.randint(1,100)

示例二:

python 复制代码
# 导入模块exceptions
from requests import exceptions as m
python 复制代码
# 导入模块exceptions
from importlib import import_module
m = import_module("requests.exceptions")

示例三:

python 复制代码
# 导入模块exceptions,获取exceptions中的InvalidURL类。
from requests.exceptions import InvalidURL
python 复制代码
# 错误方式
from importlib import import_module
m = import_module("requests.exceptions.InvalidURL") # 报错,import_module只能导入到模块级别。
python 复制代码
# 导入模块
from importlib import import_module
m = import_module("requests.exceptions")
# 去模块中获取类
cls = m.InvalidURL

在很多项目的源码中都会有 import_modulegetattr 配合实现根据字符串的形式导入模块并获取成员,例如:

python 复制代码
from importlib import import_module

path = "openpyxl.utils.exceptions.InvalidFileException"

module_path,class_name = path.rsplit(".",maxsplit=1) # "openpyxl.utils.exceptions"   "InvalidFileException"

module_object = import_module(module_path)

cls = getattr(module_object,class_name)

print(cls)

我们在开发中也可以基于这个来进行开发,提高代码的可扩展性

相关推荐
leaves falling2 小时前
C语言内存函数-
c语言·开发语言
程序员:钧念2 小时前
深度学习与强化学习的区别
人工智能·python·深度学习·算法·transformer·rag
数据与后端架构提升之路2 小时前
TeleTron 源码揭秘:如何用适配器模式“无缝魔改” Megatron-Core?
人工智能·python·适配器模式
至为芯3 小时前
IP6537至为芯支持双C口快充输出的45W降压SOC芯片
c语言·开发语言
hele_two3 小时前
快速幂算法
c++·python·算法
小羊羊Python4 小时前
SoundMaze v1.0.1正式发布!
开发语言·c++
浩瀚地学4 小时前
【Java】JDK8的一些新特性
java·开发语言·经验分享·笔记·学习
l1t4 小时前
利用DeepSeek将python DLX求解数独程序格式化并改成3.x版本
开发语言·python·算法·数独
yugi9878385 小时前
基于遗传算法优化主动悬架模糊控制的Matlab实现
开发语言·matlab
moxiaoran57536 小时前
Go语言的错误处理
开发语言·后端·golang