反射就是通过字符串的形式,导入模块;通过字符串的形式,去模块寻找指定函数,并执行。利用字符串的形式去对象(模块)中操作(查找/获取/删除/添加)成员,一种基于字符串的事件驱动!
通过import模块,我们可以引用其他文件中定义的方法,变量
思考:是否可以通过模块名动态导入
fun1.py """fun1.py模块中有三个函数""" def f1(): print('f1') def f2(): print('f2') def f3(): print('f3')
fun2.py CC = __import__("fun1")# __import__("fun1")等价于:import fun1 CC.f1() # 执行模块中的f1方法,输出f1
__import__("fun1"")相当于import fun1。导入模块。上面我们就实现了,动态导入一个模块
上面还存在一点问题:那就是我的模块名有可能不是在本级目录中存放着
那么这种方式我们该如何搞定呢?
dd = __import__("lib.text.commons") #这样仅仅导入了lib模块 dd = __import__("lib.text.commons",fromlist = True) #改用这种方式就能导入成功 inp_func = input("请输入要执行的函数:") f = getattr(dd,inp_func) f()
getattr(instance,"name","提示信息"):(找方法、属性)
class A: def __init__(self): self.name='AAAAA' def f1(self): print('f1') def f2(self): print('f2')
if __name__ == '__main__': a=A()#实例化 print(getattr(a,'name','no find')) print(getattr(a,'age','no find')) print(getattr(a,'f1','no find')) print(getattr(a,'f3','no find'))
判断对象/类是否包含名为name的特性(属性、函数)(hasattr是通过调用getattr(ojbect, name)是否抛出异常来实现的)。返回True或者False
if __name__ == '__main__': a=A()#实例化 print(hasattr(a,'name')) print(hasattr(a,'age')) print(hasattr(a,'f1')) print(hasattr(a,'f3'))
这是相对应的getattr()。参数是一个对象/类,一个字符串和一个任意值。字符串可能会列出一个现有的属性(参数是一个对象,给原有的属性赋新的值,但是这个值得作用域只是该对象,如果实例化一个新的对象,那该属性的新值将不起作用)或一个新的属性(给对象添加一个新的属性,但只能是该对象使用,如果实例化一个新的对象,那该属性将不起作用)。这个函数将值赋给属性的。该对象允许它提供。
例如:setattr(x,“foobar”,123)相当于x.foobar = 123。
class A: def __init__(self): self.name='AAAAA' def f1(self): print('f1') def f2(self): print('f2')
if __name__ == '__main__': a=A()#实例化 setattr(a,'name',23)#相当于a.name=23 print(getattr(a,'name','no find')) setattr(a,'age','asdaw')#相当于a.age=12 print(getattr(a,'age','no find')) b=A() print(getattr(b,'age','no find'))#对象b将不会拥有age属性 print(getattr(b,'name','no find'))#对象b也不会得到a修改后的值
与setattr()相关的一组函数。参数是由一个对象(记住python中一切皆是对象)和一个字符串组成的。string参数必须是对象属性名之一。该函数删除该obj的一个由string指定的属性(不影响其他实例化的对象)。delattr(x, 'foobar')=del x.foobar
if __name__ == '__main__': a=A()#实例化 delattr(a,'name') print(getattr(a,'name','no find')) print(getattr(b,'name','no find'))#对象b还是有name属性