python的反射机制,核心就是利用字符串去已存在的模块中找到指定的属性或方法,找到方法后自动执行,基于字符串的事件驱动!这也是python强大的自省能力!在Django的类视图 和API开发中,路由的底层dispatch方法就是利用反射机制实现的,现在就来看看它的实现原理。
1.首先我们先创建一个views.py模块,用于实现web框架的视图函数功能:
def index():
print('这是主页面')
def registe():
print('这是注册页面')
def login():
print('这是登录页面')
2.我们创建一个manage.py模块,用于实现web框架的启动页面:
import views
def main():
url = input('请输入要访问页面的url: ').strip()
if url == 'index':
views.index()
elif url == 'login':
views.login()
elif url == 'registe':
views.registe()
else:
print('404')
if __name__ == '__main__':
main()
执行结果:
举个例子:
def func():
print('func是函数的名字!')
a = "func"
print(a,type(a))
a是一个字符串,我们可以用func()调用函数,但是不能用字符串去掉用函数func(),“func”()这样是行不通的,但是有没有一种办法让我们直接输入的字符串去调用函数呢?python是动态语言,有很强的自省能力,自省就是面向对象的语言所写的程序在运行时,能够知道对象的类型。简单一句就是,运行时能够获知对象的类型。这里就要提到四个重要的方法:
getattr() hasattr() setattr () delattr()
1.getattr(__o,name,__default):获得对象中的方法或变量的内存地址:
class Student():
def __init__(self,name):
self.name = name
def study(self):
print('%s正在学习'%self.name)
s1 = Student('小明')
a =getattr(s1,'name') #获取name变量的内存地址
print(a) #输出:小明
b = getattr(s1,'study') #获取study的内存地址,从而可以用传入的字符串去动态的调用函数
b() #调用study()方法 输出:小明正在学习
c = getattr(s1,'age','no attribute')
print(c) #输出:no attribute。因为age属性在对象s1中没有,本应该报错,但是设置参数后找不到就报 no attribute
2.hasattr(__o,__name):判断对象中是否有这个方法或变量,返回True 或False
class Student():
def __init__(self,name):
self.name = name
def study(self):
print('%s正在学习'%self.name)
s1 = Student('小明')
print(hasattr(s1,'study')) #输出True s1实例对象有study()方法
print(hasattr(s1,'name')) #输出True s1实例对象有name属性
print(hasattr(s1,'eat')) #输出False s1实例对象没有eat属性或方法
3. setattr(__object,__name,__value):为实例化对象动态添加属性或方法
def study(self):
print('%s正在学习'%self.name)
class Student():
def __init__(self,name):
self.name = name
s1 = Student('小明')
setattr(s1,'study',study)
s1.study(s1) #动态给实例化对象添加方法
setattr(s1,'age',18)
print(s1.age) #动态给实例化对象添加属性
在我们上面搭建的Web框架中我们就会发现我们输入的url就是views模块中的方法名,如果我们直接用输入得我字符串去调用对应的方法就会省好大的事,因为如果views.py模块中有很多视图函数的话,我们在manage.py中就要一个一个去判断方法名,就很麻烦,所有用getattr()一行代码就可以轻松搞定了!这也就是Django框架中类视图 API接口底层dispatch()方法分发视图函数的反射机制,代码如下:
import views
def main():
url = input('请输入要访问的url: ').strip()
run = getattr(views, url)
run()
if __name__ == '__main__':
main()
输入:index
输出:这是主页面
代码优化:上面输入中,若非法输入或传入views中没有的视图函数将报错,这时我们可以用hasattr()去判断views模块中是否有传入的视图函数,hasattr()返回True,False:代码如下
import views
def main():
url = input('请输入要访问的url: ').strip()
if hasattr(views,url):
run = getattr(views,url)
run()
else:
print('404')
if __name__ == '__main__':
main()
#这样代码就和原来的执行效果一模一样了