如何检索模块的路径?

Che*_*ery 697 python module inotify

我想检测模块是否已更改.现在,使用inotify很简单,您只需要知道要从中获取通知的目录.

如何在python中检索模块的路径?

ore*_*tis 817

import a_module
print(a_module.__file__)
Run Code Online (Sandbox Code Playgroud)

实际上会给你加载的.pyc文件的路径,至少在Mac OS X上.所以我想你可以这样做:

import os
path = os.path.abspath(a_module.__file__)
Run Code Online (Sandbox Code Playgroud)

你也可以尝试:

path = os.path.dirname(a_module.__file__)
Run Code Online (Sandbox Code Playgroud)

获取模块的目录.

  • 这回答了如何获取导入的模块的路径,而不是你所在的模块/脚本的路径(对于你正在运行的脚本,`__ file__`不是完整路径,它是相对的).对于我所在的文件,我必须从同一目录导入另一个模块,并按此处所示进行操作.有谁知道更方便的方式? (52认同)
  • @BenBryant @hbdgaf`os.path.dirname(__ file __)`对我来说很好,并返回模块目录的abs路径. (24认同)
  • 我尝试这样做并得到回溯:`AttributeError:'module'对象没有属性'__file __'` (8认同)
  • @hbdgaf非常确定没有内置的`self .__ file__`这样的东西 (4认同)
  • @DorianDore我有点搞乱模块并得到一个解决方案`path = module .__ path __.__ dict __ ["_ path"] [0]`,但我不确定它是否可移植或者它是否在python之间没有区别版本.它对我有用,不像这个答案给了我同样的错误,`inspect`回答引起了`TypeError:<module'模块'(命名空间)>是一个内置的模块`... (4认同)

Tom*_*cek 251

inspectpython中有模块.

官方文件

inspect模块提供了几个有用的函数来帮助获取有关活动对象的信息,例如模块,类,方法,函数,回溯,框架对象和代码对象.例如,它可以帮助您检查类的内容,检索方法的源代码,提取和格式化函数的参数列表,或获取显示详细回溯所需的所有信息.

例:

>>> import os
>>> import inspect
>>> inspect.getfile(os)
'/usr/lib64/python2.7/os.pyc'
>>> inspect.getfile(inspect)
'/usr/lib64/python2.7/inspect.pyc'
>>> os.path.dirname(inspect.getfile(inspect))
'/usr/lib64/python2.7'
Run Code Online (Sandbox Code Playgroud)

  • 您也可以使用`inspect`来获取当前文件的名称; 请参阅http://stackoverflow.com/a/50905/320036 (7认同)
  • 我多次用Google搜索这个问题,*这个*是我见过的最合理的答案!请更新有关`inspect.currentframe()`的信息 (5认同)

mcs*_*her 66

正如其他答案所说,做到这一点的最好方法是__file__(再次在下面说明).但是,有一个重要的警告,即__file__如果您自己运行模块(即as __main__),则不存在.

例如,假设您有两个文件(两个文件都在您的PYTHONPATH上):

#/path1/foo.py
import bar
print(bar.__file__)
Run Code Online (Sandbox Code Playgroud)

#/path2/bar.py
import os
print(os.getcwd())
print(__file__)
Run Code Online (Sandbox Code Playgroud)

运行foo.py将给出输出:

/path1        # "import bar" causes the line "print(os.getcwd())" to run
/path2/bar.py # then "print(__file__)" runs
/path2/bar.py # then the import statement finishes and "print(bar.__file__)" runs
Run Code Online (Sandbox Code Playgroud)

但是,如果您尝试单独运行bar.py,您将获得:

/path2                              # "print(os.getcwd())" still works fine
Traceback (most recent call last):  # but __file__ doesn't exist if bar.py is running as main
  File "/path2/bar.py", line 3, in <module>
    print(__file__)
NameError: name '__file__' is not defined 
Run Code Online (Sandbox Code Playgroud)

希望这可以帮助.在测试所提出的其他解决方案时,这个警告花费了我很多时间和困惑.

  • 在这种情况下,您可以使用sys.argv [0]代替__file__. (5认同)
  • 这是一个版本特定的警告吗?在2.6和2.7中,我成功地依赖__file__,它在__name __ =='__ main__'时起作用.我见过的唯一失败案例是"python -c'print __file__'".我将补充一点,有时__file__可以是'<stdin>',这在像emacs这样的IDE执行当前缓冲区时会发生. (4认同)
  • 如何获得“NameError”?@Paul Du Bois:我尝试过Python 2.3-3.4,并且定义了`__file__`,但是我运行Python文件:`python a.py`、`python -ma`、`./a.py`。 (2认同)

jpg*_*eek 38

我会尝试解决这个问题的一些变化:

  1. 找到被调用脚本的路径
  2. 找到当前正在执行的脚本的路径
  3. 找到被调用脚本的目录

(其中一些问题已在SO上提出,但已被重复关闭并重定向到这里.)

使用注意事项 __file__

对于已导入的模块:

import something
something.__file__ 
Run Code Online (Sandbox Code Playgroud)

将返回模块的绝对路径.但是,给定下面的脚本foo.py:

#foo.py
print '__file__', __file__
Run Code Online (Sandbox Code Playgroud)

用'python foo.py'调用它将返回'foo.py'.如果你添加一个shebang:

#!/usr/bin/python 
#foo.py
print '__file__', __file__
Run Code Online (Sandbox Code Playgroud)

并使用./foo.py调用它,它将返回'./foo.py'.从不同的目录调用它(例如将foo.py放在目录栏中),然后调用其中一个

python bar/foo.py
Run Code Online (Sandbox Code Playgroud)

或添加shebang并直接执行文件:

bar/foo.py
Run Code Online (Sandbox Code Playgroud)

将返回'bar/foo.py'(相对路径).

查找目录

现在从那里去获取目录,os.path.dirname(__file__)也可能很棘手.至少在我的系统上,如果从与文件相同的目录中调用它,它将返回一个空字符串.恩.

# foo.py
import os
print '__file__ is:', __file__
print 'os.path.dirname(__file__) is:', os.path.dirname(__file__)
Run Code Online (Sandbox Code Playgroud)

将输出:

__file__ is: foo.py
os.path.dirname(__file__) is: 
Run Code Online (Sandbox Code Playgroud)

换句话说,它返回一个空字符串,因此如果要将其用于当前文件(而不是导入模块的文件),这似乎不可靠.要解决这个问题,您可以将其包装在对abspath的调用中:

# foo.py
import os
print 'os.path.abspath(__file__) is:', os.path.abspath(__file__)
print 'os.path.dirname(os.path.abspath(__file__)) is:', os.path.dirname(os.path.abspath(__file__))
Run Code Online (Sandbox Code Playgroud)

输出如下:

os.path.abspath(__file__) is: /home/user/bar/foo.py
os.path.dirname(os.path.abspath(__file__)) is: /home/user/bar
Run Code Online (Sandbox Code Playgroud)

请注意,abspath()不解析符号链接.如果要执行此操作,请改用realpath().例如,使用以下内容创建指向file_import_testing.py的符号链接file_import_testing_link:

import os
print 'abspath(__file__)',os.path.abspath(__file__)
print 'realpath(__file__)',os.path.realpath(__file__)
Run Code Online (Sandbox Code Playgroud)

执行将打印绝对路径,如:

abspath(__file__) /home/user/file_test_link
realpath(__file__) /home/user/file_test.py
Run Code Online (Sandbox Code Playgroud)

file_import_testing_link - > file_import_testing.py

使用检查

@SummerBreeze提到使用inspect模块.

对于导入的模块,这似乎运行良好,而且非常简洁:

import os
import inspect
print 'inspect.getfile(os) is:', inspect.getfile(os)
Run Code Online (Sandbox Code Playgroud)

顺从地返回绝对路径.但是,为了找到当前正在执行的脚本的路径,我没有看到使用它的方法.

  • 您可以使用`inspect.getfile(inspect.currentframe())`来获取当前运行脚本的路径. (4认同)
  • 这应该是公认的答案。感谢您的详尽回答。 (2认同)

Pla*_*ong 29

我不明白为什么没有人在谈论这个,但对我来说最简单的解决方案是使用imp.find_module("modulename")(这里的文档):

import imp
imp.find_module("os")
Run Code Online (Sandbox Code Playgroud)

它给出了一个元组,其路径位于第二位:

(<open file '/usr/lib/python2.7/os.py', mode 'U' at 0x7f44528d7540>,
'/usr/lib/python2.7/os.py',
('.py', 'U', 1))
Run Code Online (Sandbox Code Playgroud)

这种方法优于"检查"方法的优点是您不需要导入模块使其工作,您可以在输入中使用字符串.例如,检查在另一个脚本中调用的模块时很有用.

编辑:

在python3中,importlib模块应该:

文件importlib.util.find_spec:

返回指定模块的规范.

首先,检查sys.modules以查看模块是否已导入.如果是,则sys.modules [name].spec返回.如果碰巧将其设置为None,则会引发ValueError.如果模块不在sys.modules中,则会在sys.meta_path中搜索合适的规范,并为查找程序提供值"path".如果找不到规范,则返回None.

如果名称用于子模块(包含点),则会自动导入父模块.

name和package参数与importlib.import_module()的作用相同.换句话说,相对模块名称(带有前导点)起作用.

  • 一个原因是`imp`是[已弃用](http://bugs.python.org/issue17177). (4认同)
  • imp不在python 2中折旧(当前版本2.7.13).从3.4开始,imp在python 3中折旧.importlib应该在python 3中使用.我喜欢这个解决方案,因为它甚至可以在实际导入失败时工作(例如因为32位引擎的64位模块) (2认同)
  • importlib.machinery.PathFinder()。find_module(“ os”)。get_filename()`我在Python3 +中找到了`imp.find_module`的最短替代方法。如果有人在使用importlib.util.find_spec的用法。 (2认同)

Che*_*ery 19

这是微不足道的.

每个模块都有一个__file__变量,显示您现在所处的相对路径.

因此,获取模块的目录以通知它很简单:

os.path.dirname(__file__)
Run Code Online (Sandbox Code Playgroud)

  • 几乎但不太正确 - __file__*不是*"相对于你现在的位置"; 当它是相对的(只有当sys.path中有相对路径时才会这样),它相对于*模块加载时的位置*. (13认同)
  • 从 Python 3.9 开始 `__file__` 是[始终是绝对路径](https://docs.python.org/3.9/whatsnew/3.9.html#other-language-changes)。 (3认同)

小智 17

import os
path = os.path.abspath(__file__)
dir_path = os.path.dirname(path)
Run Code Online (Sandbox Code Playgroud)


Bai*_*ong 14

如果您想检索模块路径而不加载它:

import importlib.util

print(importlib.util.find_spec("requests").origin)
Run Code Online (Sandbox Code Playgroud)

输出示例:

/usr/lib64/python3.9/site-packages/requests/__init__.py
Run Code Online (Sandbox Code Playgroud)


Jos*_*ush 9

命令行实用程序

您可以将其调整为命令行实用程序,

python-which <package name>
Run Code Online (Sandbox Code Playgroud)

在此输入图像描述


创建 /usr/local/bin/python-which

#!/usr/bin/env python

import importlib
import os
import sys

args = sys.argv[1:]
if len(args) > 0:
    module = importlib.import_module(args[0])
    print os.path.dirname(module.__file__)
Run Code Online (Sandbox Code Playgroud)

让它可执行

sudo chmod +x /usr/local/bin/python-which
Run Code Online (Sandbox Code Playgroud)


Luk*_*kas 8

import module
print module.__path__
Run Code Online (Sandbox Code Playgroud)

包支持另一个特殊属性__path__.这被初始化为一个列表,其中包含在__init__.py执行该文件中的代码之前保存包的目录的名称.这个变量可以修改; 这样做会影响将来对包中包含的模块和子包的搜索.

虽然通常不需要此功能,但它可用于扩展程序包中的模块集.

资源


uri*_*uri 7

所以我花了相当多的时间尝试用py2exe做这个问题是获取脚本的基本文件夹,无论它是作为python脚本运行还是作为py2exe可执行文件运行.无论是从当前文件夹,另一个文件夹或(这是最难的)从系统的路径运行它也能使它工作.

最后我使用了这种方法,使用sys.frozen作为在py2exe中运行的指示器:

import os,sys
if hasattr(sys,'frozen'): # only when running in py2exe this exists
    base = sys.prefix
else: # otherwise this is a regular python script
    base = os.path.dirname(os.path.realpath(__file__))
Run Code Online (Sandbox Code Playgroud)


Y0d*_*0da 7

当您导入模块时,您可以访问大量信息。查看dir(a_module)。至于路径,有一个dunder a_module.__path__:。您也可以只打印模块本身。

>>> import a_module
>>> print(dir(a_module))
['__builtins__', '__cached__', '__doc__', '__file__', '__loader__', '__name__', '__package__', '__path__', '__spec__']
>>> print(a_module.__path__)
['/.../.../a_module']
>>> print(a_module)
<module 'a_module' from '/.../.../a_module/__init__.py'>
Run Code Online (Sandbox Code Playgroud)


fr_*_*lio 6

如果你想从它的任何模块中检索包的根路径,以下工作(在 Python 3.6 上测试):

from . import __path__ as ROOT_PATH
print(ROOT_PATH)
Run Code Online (Sandbox Code Playgroud)

__init__.py也可以使用__file__代替来引用主路径。

希望这可以帮助!


mik*_*ent 6

呃……

>>> import logging
>>> print(logging)
<module 'logging' from 'C:\\Users\\Mike\\AppData\\Local\\Programs\\Python\\Python39\\lib\\logging\\__init__.py'>
Run Code Online (Sandbox Code Playgroud)


小智 5

您可以导入模块,然后点击其名称,然后获取其完整路径

>>> import os
>>> os
<module 'os' from 'C:\\Users\\Hassan Ashraf\\AppData\\Local\\Programs\\Python\\Python36-32\\lib\\os.py'>
>>>
Run Code Online (Sandbox Code Playgroud)


Vla*_*den 5

如果您想知道脚本中的绝对路径,您可以使用Path对象:

from pathlib import Path

print(Path().absolute())
print(Path().resolve('.'))
print(Path().cwd())
Run Code Online (Sandbox Code Playgroud)

cwd() 方法

返回一个表示当前目录的新路径对象(由 os.getcwd() 返回)

解决()方法

使路径成为绝对路径,解析任何符号链接。返回一个新的路径对象:


Jav*_*avi 5

如果您使用 pip 安装它,“pip show”效果很好(“位置”)

$ pip 显示 detector2

Name: detectron2
Version: 0.1
Summary: Detectron2 is FAIR next-generation research platform for object detection and segmentation.
Home-page: https://github.com/facebookresearch/detectron2
Author: FAIR
Author-email: None
License: UNKNOWN
Location: /home/ubuntu/anaconda3/envs/pytorch_p36/lib/python3.6/site-packages
Requires: yacs, tabulate, tqdm, pydot, tensorboard, Pillow, termcolor, future, cloudpickle, matplotlib, fvcore
Run Code Online (Sandbox Code Playgroud)

更新:

$ python -m pip 显示我的模块

(作者:维斯巴基)