>  기사  >  백엔드 개발  >  Python 모듈 검색의 원리와 방법 소개

Python 모듈 검색의 원리와 방법 소개

巴扎黑
巴扎黑원래의
2017-08-12 14:31:061506검색

이 글은 주로 Python에서 모듈 검색의 원리와 방법을 소개합니다. 이 글에서는 예제 코드를 통해 매우 자세하게 소개합니다. 필요한 모든 사람의 학습이나 학습에 도움이 되는 내용입니다. 그리고 공부하세요.

머리말

이 글은 주로 Python 모듈 검색의 원리와 방법을 소개하고 참고 및 학습을 위해 공유합니다. 자세한 소개는

에서 살펴보겠습니다. 기본 개념

module

Module, py 파일 또는 가져올 수 있는 기타 파일은 module

package

package, __init__ 파일

상대 경로

을 포함하는 폴더입니다. 상대 경로, 디렉터리에 대한 상대 경로

절대 경로

절대 경로, 전체 경로

경로 검색

python 인터프리터가 가져온 패키지 또는 모듈을 찾습니다

Python 인터프리터가 패키지를 검색하는 방법 그리고 module

Python은 py 파일을 실행합니다. 실행 방법이 절대 경로인지 상대 경로인지에 관계없이 인터프리터는 파일이 있는 디렉터리를 sys.path에 추가합니다. 목록에서 Python은 sys.path에서 패키지와 모듈을 검색하고 sys.path 자체의 콘텐츠는 Python의 환경 변수에 의해 결정됩니다. sys.path 这个 list 中,Python 就是在 sys.path 中查找包和模块的,sys.path 中的内容本身又是又 Python 的环境变量决定。

code-1


#test.py
import os
import sys
print sys.path[0]
# execute
python test.py
python /Users/x/workspace/blog-code/p2016_05_28_python_path_find/test.py

执行表明相对路径和绝对路径都输出相同的结果,而且无论哪种执行方式,test.py 所在的文件夹都会被加入 sys.path 的首位,也就是索引为0的位置。

Python 解释器查找包的顺序是什么

解释器查找包,首先搜索 built-in module,其次搜索 sys.path ,这样的查找顺序将会导致同名包或模块被遮蔽。

code-2


#ls
├── os.py
├── test2.py
├── redis.py
#test2.py
import os
from redis import Redis
#execute test2.py
Traceback (most recent call last):
 File "/Users/x/workspace/blog-code/p2016_05_28_python_path_find/test2.py", line 1, in <module>
 from redis import Redis
ImportError: cannot import name Redis

由于 os 是 built-in module,即使在同目录下有同名模块,解释器依然可以找到正确的 os 模块,可以证实 built-in module 不会被遮蔽,而 redis 属于第三方模块,默认安装位置是 Python 环境变量中的 site-packages,解释器启动之后会将此目录中的内容加入 sys.path,由于当前目录会在 sys.path 的首位,当前目录的 redis 优先被找到,site-packages 中的 redis 模块被遮蔽了。

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

进入交互式执行环境,解释器会自动把当前目录加入 sys.path, 这时当前目录是以相对路径的形式出现在 sys.path 中:


>>> import os.path
>>> import sys
>>> os.path.abspath(sys.path[0])
&#39;/Users/x/workspace/blog-code&#39;
>>>

除此之外,其他与执行一个文件是相同的。

模块中的 __file__ 变量

__file__ is the pathname of the file from which the module was loaded, if it was loaded from a file. 如果一个模块是从文件加载的,__file__ 就是该模块的路径名–Python Doc:

顾名思义,当模块以文件的形式出现 __file__ 指的是模块文件的路径名,以相对路径执行 __file__ 是相对路径,以绝对路径执行 __file__ 是绝对路径。


#test3.py
print __file__
#相对路径执行
python test3.py
test3.py
#绝对路径执行
python /Users/x/workspace/blog-code/p2016_05_28_python_path_find/test3.py
/Users/x/workspace/blog-code/p2016_05_28_python_path_find/test3.py

为了保证__file__ 每次都能准确得到模块的正确位置,最好对其再取一次绝对路径 os.path.abspath(__file__)

交互式 shell 中的 __file__


>>> __file__
Traceback (most recent call last):
 File "<input>", line 1, in <module>
NameError: name &#39;__file__&#39; is not defined

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

sys.argv[0] 变量

sys.argv[0] 是它用来获取主入口执行文件。


#test.py
import sys
print __file__
print sys.argv[0]

以上 print 输出相同的结果,因为主执行文件和__file__所属的模块是同一个,当我们改变入口文件,区别就出现了。


#test.py
import sys
print __file__
print sys.argv[0]
#test2.py
import test
#execute test2.py
/Users/x/workspace/blog-code/p2016_05_28_python_path_find/child/test.py #__file__
test2.py #sys.argv[0]

总的来说,sys.argv[0] 是获得入口执行文件路径,__file__ 是获得任意模块文件的路径。

sys.modules 的作用

既然 Python 是在 sys.path 中搜索模块的,那载入的模块存放在何处?答案就是 sys.modules。模块一经载入,Python 会把这个模块加入 sys.modules 中供下次载入使用,这样可以加速模块的引入,起到缓存的作用。


>>> import sys
>>> sys.modules[&#39;tornado&#39;]
Traceback (most recent call last):
 File "<input>", line 1, in <module>
KeyError: &#39;tornado&#39;
>>> import tornado
>>> sys.modules[&#39;tornado&#39;]
<module &#39;tornado&#39; from &#39;/Users/x/python_dev/lib/python2.7/site-packages/tornado/__init__.pyc&#39;>

前面说过 Python 解释器启动之后,会把预先载入 built-in module,可以通过 sys.modules

code-1


🎜
>>> sys.modules[&#39;os&#39;]
<module &#39;os&#39; from &#39;/Users/x/python_dev/lib/python2.7/os.pyc&#39;>
>>>
🎜실행해 보니 상대 경로와 절대 경로 모두 동일한 결과가 출력되고, 어떤 실행 방법을 사용해도 test.py가 있는 폴더가 에 추가됩니다. sys.path 첫 번째 위치는 인덱스가 0인 위치입니다. 🎜🎜🎜🎜Python 인터프리터가 패키지를 검색하는 순서는 무엇입니까?🎜🎜🎜🎜🎜인터프리터는 먼저 패키지를 검색한 다음 sys.path를 검색합니다. 동일한 이름을 가진 패키지나 모듈은 가려집니다. 🎜🎜code-2🎜🎜🎜🎜
>>> import os
>>> os.path.realpath(sys.modules[&#39;os&#39;].__file__)
&#39;/Users/x/python_dev/lib/python2.7/os.pyc&#39;
>>> import tornado
>>> os.path.realpath(sys.modules[&#39;tornado&#39;].__file__)
&#39;/Users/x/python_dev/lib/python2.7/site-packages/tornado/__init__.pyc&#39;
🎜os는 내장 모듈이므로 같은 디렉토리에 같은 이름의 모듈이 있어도 인터프리터는 여전히 올바른 os 모듈을 찾을 수 있습니다. 내장 모듈은 가려지지 않습니다. redis는 타사 모듈입니다. 기본 설치 위치는 Python 환경 변수의 site-packages입니다. 이 디렉터리의 내용은 sys에 추가됩니다. .path, 현재 디렉터리는 sys.path 시작 부분에 있기 때문에 현재 디렉터리의 redis가 먼저 발견되고 site-packages의 redis 모듈은 가려집니다. 🎜🎜🎜🎜대화형 실행 환경 검색 순서🎜🎜🎜🎜🎜대화형 실행 환경에 들어가면 인터프리터가 자동으로 현재 디렉터리를 sys.path에 추가하고 현재 디렉터리는 상대 디렉터리가 됩니다. path 양식은 sys.path에 나타납니다. 🎜🎜🎜🎜
def get_module_dir(name):
 path = getattr(sys.modules[name], &#39;__file__&#39;, None)
 if not path
 raise AttributeError(&#39;module %s has not attribute __file__&#39;%name)
 return os.path.dirname(os.path.abspath(path))
🎜그렇지 않으면 다른 모든 것은 파일을 실행하는 것과 같습니다. 🎜🎜🎜🎜모듈의 __file__ 변수🎜🎜🎜🎜🎜__file__은 모듈이 로드된 파일의 경로 이름입니다. 모듈이 파일에서 로드된 경우 __file__은 모듈입니다. 경로 이름 – Python Doc:🎜🎜🎜이름에서 알 수 있듯이 모듈이 파일 형태로 나타날 때 __file__은 모듈 파일의 경로 이름을 나타냅니다. __file__을 상대 경로로 실행하면 상대 경로입니다. , __file__을 절대 경로로 실행하면 절대 경로가 됩니다. 🎜🎜🎜🎜rrreee🎜 __file__이 매번 정확한 모듈 위치를 얻을 수 있도록 하려면 절대 경로 os.path.abspath(__file__)를 사용하는 것이 가장 좋습니다. 🎜🎜__file__ in the Interactive Shell🎜🎜🎜🎜rrreee🎜 현재 실행되는 Interactive Shell이 ​​파일 형태로 로드되지 않아서 __file__ 같은 속성이 없기 때문입니다. 🎜🎜🎜🎜sys.argv[0] 변수 🎜🎜🎜🎜🎜sys.argv[0] 는 기본 항목 실행 파일을 얻는 데 사용됩니다. 🎜🎜🎜🎜rrreee🎜위의 출력은 동일한 결과를 출력하는데, 이는 기본 실행 파일과 __file__이 속한 모듈이 동일하기 때문입니다. 항목 파일을 변경하면 차이점이 나타납니다. 🎜🎜🎜🎜rrreee🎜일반적으로 sys.argv[0]는 항목 실행 파일을 가져오는 경로이고, __file__은 모든 모듈 파일을 가져오는 경로입니다. 🎜🎜🎜🎜sys.modules의 역할🎜🎜🎜🎜🎜Python은 sys.path에서 모듈을 검색하므로 로드된 모듈은 어디에 저장되나요? 대답은 sys.modules입니다. 모듈이 로드되면 Python은 다음 로드를 위해 모듈을 sys.modules에 추가합니다. 이렇게 하면 모듈 도입 속도가 빨라지고 캐시 역할을 할 수 있습니다. 🎜🎜🎜🎜rrreee🎜앞서 언급했듯이 Python 인터프리터가 시작된 후 내장 모듈이 미리 로드되며 이는 sys.modules를 통해 확인할 수 있습니다. 🎜🎜🎜🎜
>>> sys.modules[&#39;os&#39;]
<module &#39;os&#39; from &#39;/Users/x/python_dev/lib/python2.7/os.pyc&#39;>
>>>

借助 sys.modules 和 __file__,可以动态获取所有已加载模块目录和路径。


>>> import os
>>> os.path.realpath(sys.modules[&#39;os&#39;].__file__)
&#39;/Users/x/python_dev/lib/python2.7/os.pyc&#39;
>>> import tornado
>>> os.path.realpath(sys.modules[&#39;tornado&#39;].__file__)
&#39;/Users/x/python_dev/lib/python2.7/site-packages/tornado/__init__.pyc&#39;


def get_module_dir(name):
 path = getattr(sys.modules[name], &#39;__file__&#39;, None)
 if not path
 raise AttributeError(&#39;module %s has not attribute __file__&#39;%name)
 return os.path.dirname(os.path.abspath(path))

summary

总的来说,Python 是通过查找 sys.path 来决定包的导入,并且系统包优先级>同目录>sys.path,Python 中的特有属性 __file__ 以及 sys.argv[0]sys.modules 都能帮助我们理解包的查找和导入概念,只要能正确理解 sys.path 的作用和行为,理解包的查找就不是难题了。

위 내용은 Python 모듈 검색의 원리와 방법 소개의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

성명:
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.