首先说说反射是什么?反射是可以以字符串的形式动态调用函数的过程。
当然反射不只是字符串的函数执行,她和exec和eval不同,反射还可以动态调用函数
eg:如def fun(): print 'a' 我们调用的时候应该是fun(),如果我们定以一个字符串b =fun,我们在以b()调用函数则会报错,所以函数调用的时候不能是字符串
但是有时候我们遇到的问题如果能以字符串读取 并且可以动态调用 就会比较方便
eg:我要定义一个网页的功能,假定下面的这些函数是在commons模块里面
def login():
print("这是一个登陆页面!")
def logout():
print("这是一个退出页面!")
def home():
print("这是网站主页面!"
这样我们输入网址,根据split分割的网址,如果网址第二项是login就调用第一个函数,是logout就调用第二个函数等等,我们可以使用if
import commons
def run():
inp = input("请输入您想访问页面的url: ").strip()
if inp == "login":
commons.login()
elif inp == "logout":
commons.logout()
elif inp == "home":
commons.home()
else:
print("404")
if __name__ == '__main__':
run()
但是如果我们的类型有100个甚至1000个的时候呢,这样如果前面我们还是if...这样我们要写100甚至1000个,那如果这个时候我们的函数可以动态读取字符串进行调用的话,我们是不是可以根据倒数第二项返回的字符串立即调用函数,不用再每个 类型使用一次if,例如:
import commons
def run():
inp = input("请输入您想访问页面的url: ").strip()
func = getattr(commons,inp)#getattr寻找commons里面的inp函数,把字符串形式变成函数类型
func()#函数执行
if __name__ == '__main__':
run()
这样是不是简单明了,直接用一个动态过程进行了函数调用
同时,我们也可以使用_import_动态调用模块,其实本身我们直接用import的 时候,里面本质的过程就是_import_(),具体使用过程如下:
def run():
inp = input("请输入您想访问页面的url: ").strip()
modules, func = inp.split("/")
obj = __import__(modules)#查找与字符串相同的模块
if hasattr(obj, func):#判断对象中是否有func,返回bool布尔值
func = getattr(obj, func)
func()
else:
print("404")
if __name__ == '__main__':
run()
另外需要 注意:如果存在模块和真实的函数调用存在出入,并且有迹可循,例如全是w.函数 这时我们直接用"w."+modules是不可以的,我们必须这样使用 __import__("lib." + modules, fromlist=True) 必须 加上fromlist参数,不然_import_就只会读取点号之前的字符串去模块中进行查找