欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

Python 面向对象之反射

程序员文章站 2022-06-27 22:14:52
Python 面向对象之反射 TOC 什么是反射? hasattr getattr setattr delattr 哪些对象可以使用反射 反射的好处 例子一 例子二 什么是反射? 程序可以访问、检查和修改它本身的状态的行为的一种能力(自省) python中在面向对象中的反射:通过字符串的形式操作对象 ......

Python 面向对象之反射

TOC

  • 什么是反射?
  • hasattr
  • getattr
  • setattr
  • delattr
  • 哪些对象可以使用反射
  • 反射的好处
  • 例子一
  • 例子二

什么是反射?

 程序可以访问、检查和修改它本身的状态的行为的一种能力(自省)
 python中在面向对象中的反射:通过字符串的形式操作对象相关属性,就是通过字符串让对象自省自检是否有字符串表示的属性。python
反射提供hasattr/getattr/setattr/delattr

hasattr

  • hasattr(obj, 'string') 让obj自省自检有没有和string名符合的属性,有返回True,没有返回False。

getattr

  • getattr(obj, 'string', None) 返回obj和string名相同的绑定方法。可以用来让一个变量引用这个返回结果,后面就可以call这个变量
    ,相当于call对象的绑定方法.如果没有,返回None,没有第三个参数则抛出异常

setattr

  • setattr(obj, 'string', val) obj设置一个和string同名的的属性,并赋值为val。

delattr

  • delattr(obj, 'string') 删除obj对象的‘string’的同名的属性。

哪些对象可以使用反射

  • python中一切都是对象,所以都可以使用反射来进行自省
  • 类可以对类的共有属性,方法(绑定方法,非绑定方法)进行反射
  • 当前模块也可以进行反射。如:判定导入的模块是否有某个方法,有的化就进行调用。

反射的好处

  • 实现可插拔机制。什么意思呢?就是可先判定某个对象(模块对象,类对象,对象等)是否有某个属性(是否插入),有则调用处理。
    没有(拔出)则走另一条逻辑。
  • 这样可以实现团队开发中,实现预定好接口,就算调用的接口没有具体完成,调用方也可以完成自己的逻辑。
  • 动态模块导入:a.import(模块名)函数 b. 使用importlib模块,使用importlib的import_module('模块名')
  • 动态导入模块使用场景:
    • a. 动态引用模块的变量,可以利用反射,切换其引用的模块,并使用模块中的属性。
    • b. 使用反射判断是否有对应属性,有则干嘛,没有则干嘛。 两种场景都是利用对象的反射来处理。核心就是利用字符串来驱动不同的事件,比如导入模块,调用函数等。
      这是一种编程方法,设计模式的提现,凝聚了高内聚、松耦合的编程思想,不能简单的用执行字符串来代替。反射和exec()和eval()不同。
      参考:http://www.cnblogs.com/yooma/p/8004788.html

例子一

class People(object):
    def __init__(self, name, age):
        self.name = name
        self.age = age

    def speak(self):
        print('%s is speaking' % self.name)


obj = People('孙悟空', 22)
act = input('悟空:').strip()
if hasattr(obj, act):
    getattr(obj, act)()
    func1 = getattr(obj, act)
    func1()
else:
    print('{} can not {}'.format(obj.name, act))

例子二

class BlackMedium(object):
    feature = 'Ugly'

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

    def sell_house(self):
        print('%s 黑中介卖房子啦!' % self.name)

    def rent_house(self):
        print('%s 黑中介租房子啦' % self.name)


sb = BlackMedium('傻逼', '美帝')
# 判定是否有属性
print(hasattr(sb, 'sell_house'))
print(hasattr(sb, 'rent_house'))
print(hasattr(sb, 'test'))

# 获取一个属性,并用一个变量引用,调用变量来执行
func1 = getattr(sb, 'sell_house', None)
func1()
func2 = getattr(sb, 'rent_house')
func2()
# func3 = getattr(sb, 'test')
# func3()


# 新增一个属性
setattr(sb, 'age', '100')
print(sb.age)
print(getattr(sb, 'age'))
setattr(sb, 'talk', lambda self: print('来买房呀%s' % self.addr))  # 这个就不是绑定方法了
sb.talk(sb)