首页 / 知识
Python 的 import 机制
2023-11-12 13:35:00
1.1什么是import机制?
通常来讲,在一段Python代码中去执行引用另一个模块中的代码,就需要使用Python的import机制。import语句是触发import机制最常用的手段,但并不是唯一手段。
importlib.import_module和__import__函数也可以用来引入其他模块的代码。
1.2import是如何执行的?
import语句会执行两步操作:
搜索需要引入的模块
将模块的名字做为变量绑定到局部变量中
搜索步骤实际上是通过__import__函数完成的,而其返回值则会作为变量被绑定到局部变量中。下面我们会详细聊到__import__函数是如果运作的。
二、import机制概览
下图是import机制的概览图。不难看出,当import机制被触发时,Python首先会去sys.modules中查找该模块是否已经被引入过,如果该模块已经被引入了,就直接调用它,否则再进行下一步。这里sys.modules可以看做是一个缓存容器。值得注意的是,如果sys.modules中对应的值是None那么就会抛出一个ModuleNotFoundError异常。下面是一个简单的实验:
In[1]:importsys
In[2]:sys.modules['os']=None
In[3]:importos
---------------------------------------------------------------------------
ModuleNotFoundErrorTraceback(mostrecentcalllast)
in
---->1importos
ModuleNotFoundError:importofoshalted;Noneinsys.modules
如果在sys.modules找到了对应的module,并且这个import是由import语句触发的,那么下一步将对把对应的变量绑定到局部变量中。
如果没有发现任何缓存,那么系统将进行一个全新的import过程。在这个过程中Python将遍历sys.meta_path来寻找是否有符合条件的元路径查找器(metapathfinder)。sys.meta_path是一个存放元路径查找器的列表。它有三个默认的查找器:
内置模块查找器
冻结模块(frozenmodule)查找器
基于路径的模块查找器。
In[1]:importsys
In[2]:sys.meta_path
Out[2]:
[_frozen_importlib.BuiltinImporter,
_frozen_importlib.FrozenImporter,
_frozen_importlib_external.PathFinder]
查找器的find_spec方法决定了该查找器是否能处理要引入的模块并返回一个ModeuleSpec对象,这个对象包含了用来加载这个模块的相关信息。如果没有合适的ModuleSpec对象返回,那么系统将查看sys.meta_path的下一个元路径查找器。如果遍历sys.meta_path都没有找到合适的元路径查找器,将抛出ModuleNotFoundError。引入一个不存在的模块就会发生这种情况,因为sys.meta_path中所有的查找器都无法处理这种情况:
In[1]:importnosuchmodule
---------------------------------------------------------------------------
ModuleNotFoundErrorTraceback(mostrecentcalllast)
in
---->1importnosuchmodule
ModuleNotFoundError:Nomodulenamed'nosuchmodule'
但是,如果这个手动添加一个可以处理这个模块的查找器,那么它也是可以被引入的:
In[1]:importsys
...:
...:fromimportlib.abcimportMetaPathFinder
...:fromimportlib.machineryimportModuleSpec
...:
...:classNoSuchModuleFinder(MetaPathFinder):
...:deffind_spec(self,fullname,path,target=None):
...:returnModuleSpec('nosuchmodule',None)
...:
...:#don'tdothisinyourscript
...:sys.meta_path=[NoSuchModuleFinder()]
...:
...:importnosuchmodule
---------------------------------------------------------------------------
ImportErrorTraceback(mostrecentcalllast)
in
11sys.meta_path=[NoSuchModuleFinder()]
12
--->13importnosuchmodule
ImportError:missingloader
可以看到,当我们告诉系统如何去find_spec的时候,是不会抛出ModuleNotFound异常的。但是要成功加载一个模块,还需要加载器loader。
加载器是ModuleSpec对象的一个属性,它决定了如何加载和执行一个模块。如果说ModuleSpec对象是“师父领进门”的话,那么加载器就是“修行在个人”了。在加载器中,你完全可以决定如何来加载以及执行一个模块。这里的决定,不仅仅是加载和执行模块本身,你甚至可以修改一个模块:
In[1]:importsys
...:fromtypesimportModuleType
...:fromimportlib.machineryimportModuleSpec
...:fromimportlib.abcimportMetaPathFinder,Loader
...:
...:classModule(ModuleType):
...:def__init__(self,name):
...:self.x=1
...:self.name=name
...:
...:classExampleLoader(Loader):
...:defcreate_module(self,spec):
...:returnModule(spec.name)
...:
...:defexec_module(self,module):
...:module.y=2
...:
...:classExampleFinder(MetaPathFinder):
...:deffind_spec(self,fullname,path,target=None):
...:returnModuleSpec('module',ExampleLoader())
...:
...:sys.meta_path=[ExampleFinder()]
In[2]:importmodule
In[3]:module
Out[3]:)>
In[4]:module.x
Out[4]:1
In[5]:module.y
Out[5]:2
从上面的例子可以看到,一个加载器通常有两个重要的方法create_module和exec_module需要实现。如果实现了exec_module方法,那么create_module则是必须的。如果这个import机制是由import语句发起的,那么create_module方法返回的模块对象对应的变量将会被绑定到当前的局部变量中。如果一个模块因此成功被加载了,那么它将被缓存到sys.modules。如果这个模块再次被加载,那么sys.modules的缓存将会被直接引用。
以上内容为大家介绍了Python的import机制,希望对大家有所帮助,如果想要了解更多Python相关知识,请关注我们
最新内容
相关内容
为何你的Python代码应是扁平与稀疏
为何你的Python代码应是扁平与稀疏的,代码,培训,信息,观察,设计,工具,嵌套,闻闻,程序员,沉思,Python之禅之所以得名,正是由于它那简明扼要的规Python 之模块重载的五种方法
Python 之模块重载的五种方法,环境,培训,方法,模块,文件夹,例子,下面,内容,语句,请看,python环境准备新建一个foo文件夹,其下包含一个bar.py文Python标准库之collections模块
Python标准库之collections模块,名字,标准,信息,电话号码,培训,工厂,位置,简介,异常,对象,collections模块简介collections是Python标准库里Python变量数据类型的转换
Python变量数据类型的转换,代码,数据,培训,信息,字符串,类型,变量,整数,浮点,函数,虽然Python是弱类型编程语言,不需要像Java或C语言那样还要Python变量的类型(弱类型语言)
Python变量的类型(弱类型语言),检测,代码,数据,培训,类型,变量,语言,赋值,数字型,字符串,在强类型的编程语言中,定义变量时要指明变量的类型,而Python模块化有哪些好处?
Python模块化有哪些好处?,管理,代码,软件,汽车,培训,复杂度,模块,功能,程序,好处,当然,如果不将程序分解成一个个独立的部分,而是整个一大坨,也Python可执行文件和模块
Python可执行文件和模块,标准,培训,模块,文件,属性,上面,内容,变量,函数,源码,python源代码文件按照功能可以分为两种类型:用于执行的可执行程Python查看模块中的属性
Python查看模块中的属性,名称,标准,环境,培训,属性,模块,对象,函数,定义,变量,内置函数dir可用于列出某模块中定义了哪些属性(全局名称空间)Python系统互动
Python系统互动,互动,系统,管理,通信,标准,管理系统,培训,操作系统,功能,进程,系统互动,主要指Python和操作系统(operatesystem)、文件系统(fipython之使用inspect模块
python之使用inspect模块,异常,信息,名字,代码,培训,资料,价值,方法,参数,模块,inspect模块提供了一系列函数用于帮助使用自省。下面仅列出较python操作系统接口模块:OS
python操作系统接口模块:OS,管理,工作,工具,系统,服务,培训,模块,文件,目录,函数,OS模块提供了很多与操作系统进行交互的函数,比如常见的使用函Python之logging模块重定向
Python之logging模块重定向,信息,标准,时间,培训,网络,实时,工程,较大,代码,地方,对于代码量较大的工程,建议使用logging模块进行输出。该模块