12

我有一个完全包含在具有以下结构的目录中的 python 程序:

myprog/
├── __init__.py
├── __main__.py
├── moduleone.py
└── moduletwo.py

我希望能够打包并分发它,以便另一个开发人员可以做pip install -e /path/to/git/clone/of/myprog,然后可以在他自己的程序中导入 myprog 并用它做一些很酷的事情。

我还希望能够在命令行中运行 myprog,如下所示:

PROMPT> python myprog

当我这样做时,我希望 python 执行__main__.py它的模块。但是,这个模块引用了一些__init__.py在命令行中运行程序和被另一个程序导入时都需要使用的函数。但是,我收到以下错误:

NameError: name 'function_you_referenced_from_init_file' is not defined

我是否必须以某种方式将这些功能导入__main__.py

我尝试了一个简单的示例,如下所示:

PROMPT> cat myprog/__init__.py
def init_myprog():
    print 'running __init__.init_myprog()'

PROMPT> cat myprog/__main__.py
import myprog
print 'hi from __main__.py'
myprog.init_myprog()

PROMPT> ls -l myprog
total 16
-rw-r--r--  1 iit    63B Aug 30 11:40 __init__.py
-rw-r--r--  1 iit    64B Aug 30 12:11 __main__.py

PROMPT> python myprog
Traceback (most recent call last):
  File "/usr/local/Cellar/python/2.7.5/Frameworks/Python.framework/Versions/2.7/lib/python2.7/runpy.py", line 162, in _run_module_as_main
    "__main__", fname, loader, pkg_name)
  File "/usr/local/Cellar/python/2.7.5/Frameworks/Python.framework/Versions/2.7/lib/python2.7/runpy.py", line 72, in _run_code
    exec code in run_globals
  File "/Users/jon/dev/myprog/__main__.py", line 1, in <module>
    import myprog
ImportError: No module named myprog
4

3 回答 3

14

__init__.py仅在您导入包加载。相反,您通过执行目录将目录视为脚本。

您仍然可以将包视为脚本,而不是目录。要同时将目录视为包主脚本,请使用-m开关:

python -m myprog
于 2013-08-30T16:42:02.273 回答
10

我是否必须以某种方式将这些功能导入__main__

是的。只有项目在builtins没有导入的情况下可用。就像是:

from myprog import func1, func2

应该做的伎俩。

如果您没有myprog在正常的 python 路径中安装,那么您可以使用以下方法解决它:

import sys
import os
path = os.path.dirname(sys.modules[__name__].__file__)
path = os.path.join(path, '..')
sys.path.insert(0, path)
from myprog import function_you_referenced_from_init_file

坦率地说,这很可怕。

我建议使用MartijnPieters 的建议并将其-m放在命令行上,在这种情况下__main__.py可能如下所示:

from myprog import function_you_referenced_from_init_file
于 2013-08-30T15:31:47.657 回答
3

因此,在查看了其他回复并没有感到兴奋之后,我尝试了:

from __init__ import *

从我的__main__.py文件中。

我想它不允许您按名称引用模块,但对于简单的情况,它似乎可以工作。

于 2015-01-20T20:29:51.833 回答