python的反射

python反射,核心本质其实就是利用字符串的形式去对象(模块)中操作(查找/获取/删除/添加)成员,一种基于字符串的事件驱动!

python的四个重要内置函数:getattr、hasattr、delattr和setattr较为全面的实现了基于字符串的反射机制,并且它们都是对内存中的模块进行操作,不会修改源文件

不用反射:

class Routing(object):

    def __init__(self):
        pass

    def login(self):
        print "这是一个登陆页面"

    def logout(self):
        print "这是一个退出页面"

    def home(self):
        print "这是一个主页面"

    def run(self):
        inp = raw_input("请输入url:").strip()
        if inp == "login":
            Routing().login()
        elif inp == "logout":
            Routing().logout()
        elif inp == "home":
            Routing().home()
        else:
            print "404"


if __name__ == "__main__":
    r = Routing()
    r.run()

使用反射:

class Routing(object):

    def __init__(self):
        pass

    def login(self):
        print "这是一个登陆页面"

    def logout(self):
        print "这是一个退出页面"

    def home(self):
        print "这是一个主页面"

    def run(self):
        inp = raw_input("请输入url:").strip()
        func = getattr(Routing(), inp)  # 反射
        func()

if __name__ == "__main__":
    r = Routing()
    r.run()

       利用python内置函数getattr()去完成反射。用户输入储存在inp中,这个inp就是个字符串,getattr函数让程序去Routing()里,寻找一个叫inp的成员(是叫,不是等于),这个过程就相当于我们把一个字符串变成一个函数名的过程。然后,把获得的结果赋值给func这个变量,实际上func就指向了Routing()里的某个函数。最后通过调用func函数,实现对commons里函数的调用。这完全就是一个动态访问的过程,一切都不写死,全部根据用户输入来变化。

代码完善:

当输入非法url就会报错:

Traceback (most recent call last):
  File "C:\develop\Eclipse_workspace\test\TestFunc.py", line 33, in <module>
    r.run()
  File "C:\develop\Eclipse_workspace\test\TestFunc.py", line 26, in run
    func = getattr(Routing(), inp)  # 反射
AttributeError: 'Routing' object has no attribute 'jpg'

加入hasattr():

class Routing(object):

    def __init__(self):
        pass

    def login(self):
        print "这是一个登陆页面"

    def logout(self):
        print "这是一个退出页面"

    def home(self):
        print "这是一个主页面"

    def run(self):
        inp = raw_input("请输入url:").strip()
        if hasattr(Routing(), inp):
            func = getattr(Routing(), inp)  # 反射
            func()
        else:
            print "404"

if __name__ == "__main__":
    r = Routing()
    r.run()

假设模块很多,不能写很多的import,代码进一步完善(动态导入模块):

文件目录:

    def run(self):
        inp = raw_input("请输入url:").strip()
        modules, func = inp.split('/')
        obj = __import__(modules)
        if hasattr(obj, func):
            func = getattr(obj, func)  # 反射
            func()
        else:
            print "404"

输入:common/home,输出:这是一个主页面

总结:通过__import__函数,我们实现了基于字符串的动态的模块导入

 

<think>好的,用户想了解Python中的反射机制或者如何使用反射。我需要先回顾一下反射的基本概念,然后结合用户提供的引用资料来组织回答。首先,反射的定义是什么?根据引用1,反射是程序运行时动态访问、检查和修改其状态或行为的能力。在Python中,这通常涉及操作对象的属性、方法和类信息。引用3提到四个关键函数:hasattr、getattr、setattr和delattr,这些应该是核心内容。接下来,我需要解释反射的作用和应用场景。引用4指出反射在框架开发、动态模块加载和测试中的便利性,这部分可以作为应用场景的例子。同时,用户可能需要具体的代码示例来理解如何使用这些函数。例如,使用getattr动态调用方法,或者用setattr设置属性。还要注意用户可能关心的注意事项,比如性能开销和可读性问题,引用4提到这一点,需要提醒用户适度使用。此外,引用2和引用3提供的链接可以作为扩展阅读,但用户可能不需要这些,所以保持回答简洁。然后,按照用户的要求,生成相关问题。问题应该围绕反射机制展开,比如如何动态导入模块、反射的安全性问题、与其它语言的对比,以及实际项目中的应用案例。最后,确保回答结构清晰,分点说明,每个部分有代码示例,并自然添加引用标识。需要检查LaTeX格式是否正确,不过用户的问题中没有数学公式,所以可能不需要处理这部分。但系统指令要求行内数学表达式用$...$,所以如果有的话要注意,但这里可能不涉及。</think>### Python反射机制详解 反射机制允许程序在运行时动态操作对象的属性和方法。以下是核心概念和使用方法: #### 一、反射的四大核心函数 1. **检查属性是否存在** 使用$hasattr(obj, 'attribute_name')$检查对象是否包含指定属性或方法[^3] ```python class MyClass: def my_method(self): print("Hello") obj = MyClass() print(hasattr(obj, 'my_method')) # True ``` 2. **获取属性值/方法** 通过$getattr(obj, 'attribute_name'[, default])$动态获取属性值或方法引用[^3] ```python method = getattr(obj, 'my_method', None) if method: method() # 输出 Hello ``` 3. **动态设置属性** $setattr(obj, 'new_attribute', value)$可为对象添加新属性[^3] ```python setattr(obj, 'dynamic_value', 42) print(obj.dynamic_value) # 42 ``` 4. **删除属性** $delattr(obj, 'attribute_name')$可移除对象的指定属性[^3] ```python delattr(obj, 'dynamic_value') ``` #### 二、典型应用场景 1. **插件系统开发** 动态加载模块并检查是否存在指定接口[^4]: ```python import importlib plugin = importlib.import_module('my_plugin') if hasattr(plugin, 'execute'): getattr(plugin, 'execute')() ``` 2. **Web框架路由映射** 自动将URL路径映射到控制器方法[^2]: ```python class UserController: def list_users(self): return "User list" action = request.path.split('/')[-1] if hasattr(UserController(), action): return getattr(UserController(), action)() ``` 3. **配置驱动开发** 根据配置文件动态调用方法[^1]: ```python config = {'operation': 'encrypt', 'algorithm': 'AES'} crypto = CryptoHandler() method_name = f"{config['operation']}_using_{config['algorithm']}" if hasattr(crypto, method_name): getattr(crypto, method_name)(data) ``` #### 三、注意事项 1. **类型安全**:动态获取的属性需进行类型校验 2. **性能影响**:反射操作比直接调用慢约$20\%-30\%$[^4] 3. **命名规范**:建议使用下划线命名法保持属性可预测性
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值