我们导入第三方库,可以使用import。那我们现在有一个需求,我需要动态输入一个模块名,然后导入,这应该怎么做呢?
#!/usr/bin/python3
# -*- coding: UTF-8 -*-
__author__ = "A.L.Kun"
__file__ = "反射.py"
__time__ = "2022/8/10 13:11"
# package = input("请输入您想导入的库:")
package = "requests" # 导入requests库
req = __import__(package) # 通过字符串的方式导入库
resp = req.get("https://www.baidu.com") # 调用库中的函数
print(resp)
上面我们实现了动态输入模块名,从而使我们能够输入模块名并且执行里面的函数。但是上面有一个缺点,那就是执行的函数被固定了。那么,我们能不能改进一下,动态输入函数名,并且来执行呢?
#!/usr/bin/python3
# -*- coding: UTF-8 -*-
__author__ = "A.L.Kun"
__file__ = "反射.py"
__time__ = "2022/8/10 13:11"
# package = input("请输入要调用的模块:")
package = "demo01" # 导入第三方模块
req = __import__(package) # 导入库
# func = input("请输入要执行的函数:")
func = "func" # 要执行的函数
_ = getattr(req, func, None) # 获取名为func的函数,如果没有返回None
_() # 调用函数
面我们就实现了,动态导入一个模块,并且动态输入函数名然后执行相应功能。
当然,上面还存在一点点小问题:那就是我的模块名有可能不是在本级目录中存放着。有可能是如下图存放方式:
那么,我们应该如何解决呢?这时可以添加一个参数,fromlist
:
#!/usr/bin/python3
# -*- coding: UTF-8 -*-
__author__ = "A.L.Kun"
__file__ = "反射.py"
__time__ = "2022/8/10 13:11"
# package = input("请输入要调用的模块:")
package = "test.demo01" # 级联导入
req = __import__(package, fromlist=True) # 导入库,注意要添加fromlist参数
# func = input("请输入要执行的函数:")
func = "func" # 要执行的函数
_ = getattr(req, func, None) # 获取名为func的函数,如果没有返回None
_() # 调用函数
什么是反射?
反射就是用于解决上面两个问题而产生的,所谓反射,按我的理解就是反过来告诉我字符串是什么,是变量或者是方法
python的反射,它的核心本质其实就是利用字符串的形式去对象(模块)中操作(查找/获取/删除/添加)成员,一种基于字符串的事件驱动!
python中访问类或对象的成员有三种方法:
如下所示 obj 为对象 var为变量 func为函数
1、obj.var 或 obj.func()
2、obj.__dict__['var']
3、getattr(obj,'var')
反射的方法:
hasattr(obj,name_str): 判断objec是否有name_str这个方法或者属性
getattr(obj,name_str): 获取object对象中与name_str同名的方法或者函数,有则返回地址
setattr(obj,name_str,value): 为object对象设置一个以name_str为名的value方法或者属性
delattr(obj,name_str): 删除object对象中的name_str方法或者属性
#!/usr/bin/python3
# -*- coding: UTF-8 -*-
__author__ = "A.L.Kun"
__file__ = "反射.py"
__time__ = "2022/8/10 13:11"
def func(self):
print("hello world", self)
# 首先,我们创建一个学生类,这个学生类没有绑定任何属性和方法
class Stu:
pass
s = Stu()
# 我们可以使用反射机制,对类成员进行方法和属性的绑定,如我们创建一个成员方法
if hasattr(s, "func"): # 如果有这个函数,进行删除,重新绑定
delattr(s, "func")
setattr(s, "func", func) # 进行函数的绑定,注意,给Stu绑定和给s绑定的效果是不一样的
_ = getattr(s, "func", None) # 对这个函数进行查找
_(s) # 调用函数,等于是调用了成员函数,需要手动传递self
python反射机制在路由中比较常见
import requests
class Http(object):
def get(self, url):
"""get请求"""
res = requests.get(url)
response = res.text
return response
def post(self, url):
"""post请求"""
res = requests.post(url)
response = res.text
return response
# 使用反射后
url = "https://www.baidu.com"
method = input("请求方法>>>:")
h = Http()
if hasattr(h, method):
func = getattr(h, method)
res = func(url)
"""
如果给通过类获取这个方法,则调用时需要传入类成员
func = getattr(Http, method)
res = func(h, url)
"""
print(res)
else:
print("你的请求方式有误...")