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)

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

相关推荐
databook40 分钟前
概率图模型:机器学习的结构化概率之道
python·机器学习·scikit-learn
拾回程序猿的圈圈∞43 分钟前
实战二:开发网页端界面完成黑白视频转为彩色视频
python·ai编程
亚林瓜子1 小时前
AWS Elastic Beanstalk + CodePipeline(Python Flask Web的国区CI/CD)
python·ci/cd·flask·web·aws·beanstalk·codepipeline
春末的南方城市1 小时前
中山大学&美团&港科大提出首个音频驱动多人对话视频生成MultiTalk,输入一个音频和提示,即可生成对应唇部、音频交互视频。
人工智能·python·深度学习·计算机视觉·transformer
深科文库1 小时前
构建 MCP 服务器:第 4 部分 — 创建工具
python·chatgpt·prompt·aigc·agi·ai-native
witton2 小时前
美化显示LLDB调试的数据结构
数据结构·python·lldb·美化·debugger·mupdf·pretty printer
SteveDraw2 小时前
C++动态链接库封装,供C#/C++ 等编程语言使用——C++动态链接库概述(总)
开发语言·c++·c#·封装·动态链接库
十五年专注C++开发2 小时前
设计模式之单例模式(二): 心得体会
开发语言·c++·单例模式·设计模式
nenchoumi31192 小时前
AirSim/Cosys-AirSim 游戏开发(一)XBox 手柄 Windows + python 连接与读取
windows·python·xbox