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)

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

相关推荐
mazo_command1 小时前
【MATLAB课设五子棋教程】(附源码)
开发语言·matlab
IT猿手1 小时前
多目标应用(一):多目标麋鹿优化算法(MOEHO)求解10个工程应用,提供完整MATLAB代码
开发语言·人工智能·算法·机器学习·matlab
青春男大1 小时前
java栈--数据结构
java·开发语言·数据结构·学习·eclipse
88号技师1 小时前
几款性能优秀的差分进化算法DE(SaDE、JADE,SHADE,LSHADE、LSHADE_SPACMA、LSHADE_EpSin)-附Matlab免费代码
开发语言·人工智能·算法·matlab·优化算法
Zer0_on1 小时前
数据结构栈和队列
c语言·开发语言·数据结构
一只小bit1 小时前
数据结构之栈,队列,树
c语言·开发语言·数据结构·c++
沐霜枫叶2 小时前
解决pycharm无法识别miniconda
ide·python·pycharm
一个没有本领的人2 小时前
win11+matlab2021a配置C-COT
c语言·开发语言·matlab·目标跟踪
途途途途2 小时前
精选9个自动化任务的Python脚本精选
数据库·python·自动化
一只自律的鸡2 小时前
C项目 天天酷跑(下篇)
c语言·开发语言