我想检测模块是否发生了变化。现在,使用inotify很简单,你只需要知道你想从哪个目录获取通知。
如何在python中检索模块的路径?
我想检测模块是否发生了变化。现在,使用inotify很简单,你只需要知道你想从哪个目录获取通知。
如何在python中检索模块的路径?
当前回答
我也会尝试解决这个问题的一些变化:
查找被调用脚本的路径 查找当前执行脚本的路径 查找被调用脚本的目录
(其中一些问题已经在SO上被问过,但在这里已经被关闭并重新定向。)
使用__file__的注意事项
对于已导入的模块:
import something
something.__file__
将返回模块的绝对路径。然而,给定下面的脚本foo.py:
#foo.py
print '__file__', __file__
用'python foo.py'调用它只会返回'foo.py'。如果你加上一个shebang:
#!/usr/bin/python
#foo.py
print '__file__', __file__
并使用。/foo.py调用它,它将返回'./foo.py'。从不同的目录调用它(例如把foo.py放在目录栏中),然后调用任何一个
python bar/foo.py
或者添加一个shebang并直接执行文件:
bar/foo.py
将返回'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__)
将输出:
__file__ is: foo.py
os.path.dirname(__file__) is:
换句话说,它返回一个空字符串,所以如果您想将它用于当前文件(而不是导入模块的文件),这似乎不可靠。为了解决这个问题,你可以调用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__))
输出如下:
os.path.abspath(__file__) is: /home/user/bar/foo.py
os.path.dirname(os.path.abspath(__file__)) is: /home/user/bar
注意,abspath()不解析符号链接。如果您想这样做,请使用realpath()。例如,创建一个符号链接file_import_testing_link指向file_import_testing.py,内容如下:
import os
print 'abspath(__file__)',os.path.abspath(__file__)
print 'realpath(__file__)',os.path.realpath(__file__)
执行将输出绝对路径,如下所示:
abspath(__file__) /home/user/file_test_link
realpath(__file__) /home/user/file_test.py
File_import_testing_link -> file_import_testing.py
使用检查
@SummerBreeze提到了使用inspect模块。
对于导入的模块,这似乎工作得很好,而且相当简洁:
import os
import inspect
print 'inspect.getfile(os) is:', inspect.getfile(os)
返回绝对路径。查找当前执行脚本的路径:
inspect.getfile(inspect.currentframe())
(感谢@jbochi)
inspect.getabsfile(inspect.currentframe())
给出了当前执行脚本的绝对路径(感谢@Sadman_Sakib)。
其他回答
如果你想从包的任何模块中检索包的根路径,下面的工作(在Python 3.6上测试):
from . import __path__ as ROOT_PATH
print(ROOT_PATH)
主__init__.py路径也可以通过使用__file__来引用。
希望这能有所帮助!
我也会尝试解决这个问题的一些变化:
查找被调用脚本的路径 查找当前执行脚本的路径 查找被调用脚本的目录
(其中一些问题已经在SO上被问过,但在这里已经被关闭并重新定向。)
使用__file__的注意事项
对于已导入的模块:
import something
something.__file__
将返回模块的绝对路径。然而,给定下面的脚本foo.py:
#foo.py
print '__file__', __file__
用'python foo.py'调用它只会返回'foo.py'。如果你加上一个shebang:
#!/usr/bin/python
#foo.py
print '__file__', __file__
并使用。/foo.py调用它,它将返回'./foo.py'。从不同的目录调用它(例如把foo.py放在目录栏中),然后调用任何一个
python bar/foo.py
或者添加一个shebang并直接执行文件:
bar/foo.py
将返回'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__)
将输出:
__file__ is: foo.py
os.path.dirname(__file__) is:
换句话说,它返回一个空字符串,所以如果您想将它用于当前文件(而不是导入模块的文件),这似乎不可靠。为了解决这个问题,你可以调用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__))
输出如下:
os.path.abspath(__file__) is: /home/user/bar/foo.py
os.path.dirname(os.path.abspath(__file__)) is: /home/user/bar
注意,abspath()不解析符号链接。如果您想这样做,请使用realpath()。例如,创建一个符号链接file_import_testing_link指向file_import_testing.py,内容如下:
import os
print 'abspath(__file__)',os.path.abspath(__file__)
print 'realpath(__file__)',os.path.realpath(__file__)
执行将输出绝对路径,如下所示:
abspath(__file__) /home/user/file_test_link
realpath(__file__) /home/user/file_test.py
File_import_testing_link -> file_import_testing.py
使用检查
@SummerBreeze提到了使用inspect模块。
对于导入的模块,这似乎工作得很好,而且相当简洁:
import os
import inspect
print 'inspect.getfile(os) is:', inspect.getfile(os)
返回绝对路径。查找当前执行脚本的路径:
inspect.getfile(inspect.currentframe())
(感谢@jbochi)
inspect.getabsfile(inspect.currentframe())
给出了当前执行脚本的绝对路径(感谢@Sadman_Sakib)。
如果你想在不加载的情况下检索模块路径:
import importlib.util
print(importlib.util.find_spec("requests").origin)
示例输出:
/usr/lib64/python3.9/site-packages/requests/__init__.py
因此,我花了相当多的时间尝试用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__))
您可以导入您的模块 然后点击它的名字,你会得到它的完整路径
>>> import os
>>> os
<module 'os' from 'C:\\Users\\Hassan Ashraf\\AppData\\Local\\Programs\\Python\\Python36-32\\lib\\os.py'>
>>>