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__函数,我们实现了基于字符串的动态的模块导入