1、介绍
反射的概念是由Smith在1982年首次提出的,主要是指程序可以访问、检测和修改它本身状态或行为的一种能力(自省)。这一概念的提出很快引发了计算机科学领域关于应用反射性的研究。它首先被程序语言的设计领域所采用,并在Lisp和面向对象方面取得了成绩。
python面向对象中的反射:通过字符串的形式操作对象相关的属性。python中的一切事物都是对象(都可以使用反射)
对类进行反射操作,执行的结果是查询或修改类的属性
对实例化对象进行反射操作,执行的结果是查询或修改实例化对象的属性
2、四个可以实现自省的函数
hasattr(对象名,'属性名') #判断对象中是否存在该属性
getattr(对象名,'属性名', 默认值可选) #获取对象中的该属性的值,可以添加默认值
setattr(对象名,'属性名', vlaue) #设置对象中的该属性名的值为vlaue
delattr(对象名,'属性名') #删除对象中的该属性
3、例子
1:对象示例(一切皆对象,类本身也是一个对象)
class Foo: def __init__(self, name, age): self.name = name self.age = age def say_hi(self): print('hi,%s'%self.name) #检测是否含有某属性 obj = Foo('Tom', 25) print(hasattr(obj, 'name')) # True print(hasattr(obj, 'say_hi')) # True #获取属性 name = getattr(obj, 'name') print(name) # Tom print(getattr(obj, 'name1', '不存在')) # 不存在 #如果不加默认值就会报错 func = getattr(obj, 'say_hi') if func: func() # hi, Tom #设置属性 setattr(obj, 'sex', 'man') print(obj.sex) # man #删除属性 delattr(obj, 'sex') print(obj.__dict__) {'name': 'Tom', 'age': 25} delattr(obj, 'name1') #不存在,则报错,AttributeError: name1
例子2:对类的示例
class Foo(object): staticField = "old boy" def __init__(self): self.name = 'wupeiqi' def func(self): return 'func' @staticmethod def bar(): return 'bar' print(getattr(Foo, 'staticField')) #old boy ret1 = getattr(Foo, 'func') print(ret1(1)) #func 传入参数1,并执行ret1获取的方法 print(getattr(Foo, 'func')(1)) #func print(getattr(Foo, 'bar')()) #bar
例子3:对当前模块的示例
import sys def s1(): print('s1') def s2(): print('s2') this_module = sys.modules[__name__] print(hasattr(this_module, 's1')) # True getattr(this_module, 's2')() # s2
例子4:其他模块的示例
#一个模块中的代码 def test(): print('from the test') """ 程序目录: module_test.py index.py 当前文件: index.py """ # 另一个模块中的代码 fs.py文件有如下代码 n1 = 'Tom' def func(): print(666) class A: name = 'Mike' def func2(self): print('--------IN FUNC2') test.py文件有如下代码: import fs print(getattr(fs, 'n1')) # Tom 可以从导入的fs模块中获取到n1属性 ret1 = getattr(fs, 'func') ret1() # 666 可以从导入的fs模块中获取func方法并执行 ret2 = getattr(fs, 'A') print(ret2.name) # Mike 可以从导入的fs模块中获取A类,打印A类的name属性 print(getattr(fs.A, 'name')) # Mike 可以从导入的fs模块中的A属性中获取name属性并执行 ret3 = getattr(fs.A, 'func2')(1) # --------IN FUNC2 获取func2方法,并传入参数1执行