众所周知import是用来加载Python模块的,其实import是调用内建函数__import__来工作的,这就使我们动态加载模块变成了可能。
import glob, os
modules = []
for module_file in glob.glob('*-plugin.py'): #glob.glob得到当前目录下匹配字符串的文件名
module_name, ext = os.path.splitext(os.path.basename(module_file)) #将文件名以点号分开
print(os.path.basename(module_file))
print(ext)
module = __import__(module_name) #获得模块
fun = getattr(module, 'hello') #根据函数名获得函数
print(fun)
fun()
modules.append(module)
for module in modules:
module.hello()
下面我们来实现一个动态加载的类,当第一次使用该模块时,才会被加载
class LazyImport:
def __init__(self, module_name):
self.module_name = module_name
self.module = None
def __getattr__(self, funcname):
if self.module is None:
self.module = __import__(self.module_name)
print(self.module)
return getattr(self.module, funcname)
string = LazyImport('string')
print(string.ascii_lowercase)
其中,这个类的__getattr__方法,只有当这个类的对象调用了某个方法,并且找不到这个方法的时候会被自动调用。
一般象a.b这样的形式,python可能会先查找a.__dict__中是否存在,如果不存在会在类的__dict__中去查找,再没找到可能会去按这种方法去父类中进行查找。实在是找不到,会调用__getattr__,如果不存在则返回一个异常。那么__getattr__只有当找不到某个属性的时候才会被调用。