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)

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

相关推荐
巴里巴气14 分钟前
selenium基础知识 和 模拟登录selenium版本
爬虫·python·selenium·爬虫模拟登录
198918 分钟前
【零基础学AI】第26讲:循环神经网络(RNN)与LSTM - 文本生成
人工智能·python·rnn·神经网络·机器学习·tensorflow·lstm
JavaEdge在掘金24 分钟前
Redis 数据倾斜?别慌!从成因到解决方案,一文帮你搞定
python
ansurfen28 分钟前
我的第一个AI项目:从零搭建RAG知识库的踩坑之旅
python·llm
Thomas_YXQ30 分钟前
Unity URP法线贴图实现教程
开发语言·unity·性能优化·游戏引擎·unity3d·贴图·单一职责原则
前端付豪32 分钟前
20、用 Python + API 打造终端天气预报工具(支持城市查询、天气图标、美化输出🧊
后端·python
前端付豪38 分钟前
19、用 Python + OpenAI 构建一个命令行 AI 问答助手
后端·python
Zz_waiting.41 分钟前
Javaweb - 10.4 ServletConfig 和 ServletContext
java·开发语言·前端·servlet·servletconfig·servletcontext·域对象
Touper.1 小时前
JavaSE -- 泛型详细介绍
java·开发语言·算法
sun0077001 小时前
std::forward作用
开发语言·c++·算法