​Python 模块的加载顺序​

基本概念

module

模块, 一个 py 文件或以其他文件形式存在的可被导入的就是一个模块

package

包,包含有 init 文件的文件夹

relative path

相对路径,相对于某个目录的路径

absolute path

绝对路径,全路径

Python 解释器是如何查找包和模块的

Python 执行一个 py 文件,无论执行的方式是用绝对路径还是相对路径,interpreter 都会把文件所在的 directory 加入 sys.path 这个 list 中,并且是索引为 0 的位置。Python 就是在 sys.path 中查找包和模块的。

[root@localhost python_test]# cat sys_test.py 
# coding:utf-8
import sys
print(sys.path)
print('Now in main.py')


def hello():
    print('michael hello')


if __name__ == '__main__':
    hello()

[root@localhost python_test]# python sys_test.py 
['/root/python_test', '/usr/local/python3.6/lib/python36.zip', '/usr/local/python3.6/lib/python3.6', '/usr/local/python3.6/lib/python3.6/lib-dynload', '/usr/local/python3.6/lib/python3.6/site-packages']
Now in main.py
michael hello

Python 解释器查找包的顺序

解释器查找包

  • 解释器会默认加载一些 modules,除了sys.builtin_module_names 列出的内置模块之外,还会加载其他一些标准库,都存放在sys.modules字典中。
  • 然后就是搜索 sys.path 路径下的模块了。

>>> import sys
>>> print(sys.builtin_module_names)
('_ast', '_codecs', '_collections', '_functools', '_imp', '_io', '_locale', '_operator', '_signal', '_sre', '_stat', '_string', '_symtable', '_thread', '_tracemalloc', '_warnings', '_weakref', 'atexit', 'builtins', 'errno', 'faulthandler', 'gc', 'itertools', 'marshal', 'posix', 'pwd', 'sys', 'time', 'xxsubtype', 'zipimport')

这样的查找顺序将会导致同名包或模块被遮蔽。

示例2:

[root@localhost package]# tree . -L 1
.
├── __init__.py
├── name
├── os.py
├── test2.py
├── test.py
└── test.pyc

[root@localhost package]# cat test2.py 
import os
from redis import Redis
from test import hello

print('Now in test2.py')
print(os.getcwd())
[root@localhost package]# python test2.py 
Traceback (most recent call last):
  File "test2.py", line 2, in <module>
    from redis import Redis
ModuleNotFoundError: No module named 'redis'

这里的 os 模块并不是 built-in module,上面已经将 sys.builtin_module_names 内容打印出来了。只是 Python 解释器启动时就加载到了 sys.modules中缓存起来了。所以,即使在同目录下有同名模块,解释器依然是可以找到正确的 os 模块的!如果你在import os之前,先执行del sys.modules['os'],那么,标准模块 os 就会被同目录下的 os.py 屏蔽了。

redis 属于第三方模块,默认安装位置是 Python 环境变量中的 site-packages,解释器启动之后,会将此目录加到 sys.path,由于当前目录会在 sys.path 的首位,如果当前目录的 redis 优先被找到了,那么site-packages 中的 redis 模块就会被屏蔽。

综上所述,搜索的一个顺序是:

sys.modules 缓存 -> sys.path[0] 即当前目录查找 -> sys.path[1:]路径查找

同时发现,模块被加载的时候,其中非函数或类的语句,例如 print('hello')name=michael等,是会在 import的时候,默认就执行了。

交互式执行环境的查找顺序

交互执行环境,解释器会自动把当前目录加入到sys.path,这一点和直接执行文件是一样的,但是这种方式下,sys.path[0] 是存储的当前目录的相对路径,而不是绝对路径。

>>> import sys
>>> sys.path[0]
''

模块中的 __file__ 变量

文件中的 __file__

当模块以文件的形式出现 file 指的是模块文件的路径名,以相对路径执行 file 是相对路径,以绝对路径执行 file 是绝对路径:

[root@localhost package]# cat test3.py 
print(__file__)
[root@localhost package]# python test3.py
test3.py
[root@localhost package]# pwd
/root/python_test/package
[root@localhost package]# python /root/python_test/package/test3.py
/root/python_test/package/test3.py

交互式 Shell 中的 __file__

交互式 Shell 的执行并不是以文件的形式加载,所以不存在 __file__ 这样的属性。

>>> __file__
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
NameError: name '__file__' is not defined

sys.argv[0] 变量

sys.argv[0] 是获得入口执行文件路径,__file__ 是真实被执行模块的文件路径。比如下面例子中,test2.py 就是入口执行文件,而 test.py 就是在 import 时真实被执行的模块。

[root@localhost package]# cat test.py
import sys


print(__file__)
print(sys.argv[0])
[root@localhost package]# cat test1.py 
import test

[root@localhost package]# python test1.py 
/root/python_test/package/test.py
test1.py

sys.modules 的作用

载入的模块存放在何处?答案是 sys.modules。模块一经载入,Python 会把这个模块加入 sys.modules 中供下次载入使用,这样可以加速模块引入,起到缓存的作用。sys.modules 是一个 dict 类型的值。

>>> sys.modules['requests']
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
KeyError: 'requests'
>>> import requests
>>> sys.modules['requests']
<module 'requests' from '/usr/local/python3.6/lib/python3.6/site-packages/requests/__init__.py'>

# 没有预先引入 math,但是 sys.modules 中已经有这个键

>>> sys.modules['math']
<module 'math' from '/usr/local/python3.6/lib/python3.6/lib-dynload/math.cpython-36m-x86_64-linux-gnu.so'>

需要注意的是,sys.modules['math']尽管可以看到 math 键,但是要使用它,还是需要显式的引入之后才能使用,因为那只是Python解释器后台缓存的,不显示引入,本地空间还是不会去发现它。

总结

Python 通过查找 sys.path 来决定包的导入,Python解释器启动时加载的模块缓存 > 同级目录 > sys.path[1:]

Python 中的特有属性 __file__ 以及 sys.argv[0]sys.argv[0]sys.modules 可以帮助分析包的查找和导入过程。

参考

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值