我试图遵循PEP 328,其目录结构如下:

pkg/
  __init__.py
  components/
    core.py
    __init__.py
  tests/
    core_test.py
    __init__.py

在core_test.py中,我有以下import语句

from ..components.core import GameLoopEvents

然而,当我运行时,我得到以下错误:

tests$ python core_test.py 
Traceback (most recent call last):
  File "core_test.py", line 3, in <module>
    from ..components.core import GameLoopEvents
ValueError: Attempted relative import in non-package

四处搜索,我发现“相对路径即使使用__init__.py也无法工作”和“从相对路径导入模块”,但它们都没有帮助。

这里有什么我遗漏的吗?


当前回答

在core_test.py中,执行以下操作:

import sys
sys.path.append('../components')
from core import GameLoopEvents

其他回答

正如Paolo所说,我们有两种调用方法:

1) python -m tests.core_test
2) python tests/core_test.py

它们之间的一个区别是sys.path[0]字符串。由于在执行导入时解释将搜索sys.path,因此我们可以使用tests/core_test.py:

if __name__ == '__main__':
    import sys
    from pathlib import Path
    sys.path.insert(0, str(Path(__file__).resolve().parent.parent))
    from components import core
    <other stuff>

之后,我们可以使用其他方法运行core_test.py:

cd tests
python core_test.py
python -m core_test
...

注意,仅测试了py36。

这取决于您希望如何启动脚本。

如果您想以经典的方式从命令行启动UnitTest,即:

python tests/core_test.py

然后,因为在这种情况下,“components”和“tests”是同级文件夹,所以可以使用sys.path模块的insert或append方法导入相关模块。类似于:

import sys
from os import path
sys.path.append( path.dirname( path.dirname( path.abspath(__file__) ) ) )
from components.core import GameLoopEvents

否则,您可以使用“-m”参数启动脚本(请注意,在本例中,我们讨论的是一个包,因此您不能使用“.py”扩展名),即:

python -m pkg.tests.core_test

在这种情况下,您可以简单地使用相对导入:

from ..components.core import GameLoopEvents

您最终可以将这两种方法混合使用,这样无论如何调用脚本,脚本都能正常工作。例如:

if __name__ == '__main__':
    if __package__ is None:
        import sys
        from os import path
        sys.path.append( path.dirname( path.dirname( path.abspath(__file__) ) ) )
        from components.core import GameLoopEvents
    else:
        from ..components.core import GameLoopEvents

对你没有把它当作一个包。

python -m pkg.tests.core_test

如果您的用例是用于运行测试的,并且与之吻合,那么您可以执行以下操作。不要将测试脚本作为python core_test.py运行,而是使用诸如pytest之类的测试框架。然后在命令行上输入

$$ py.test

这将在您的目录中运行测试。这解决了@BrenBarn指出的__name__是__main__的问题。接下来,将一个空__init__.py文件放入测试目录,这将使测试目录成为包的一部分。那你就可以

from ..components.core import GameLoopEvents

然而,如果您将测试脚本作为主程序运行,那么事情将再次失败。所以只需要使用测试运行器。也许这也适用于其他测试者,比如鼻测试,但我没有检查过。希望这有帮助。

问题在于您的测试方法,

您尝试了python core_test.py

那么你会得到这个错误ValueError:尝试在非包中进行相对导入

原因:您正在从非包源测试您的包。

所以从包源测试模块。

如果这是您的项目结构,

pkg/
  __init__.py
  components/
    core.py
    __init__.py
  tests/
    core_test.py
    __init__.py

cd软件包

python -m tests.core_test # dont use .py

或从外部pkg/

python -m pkg.tests.core_test

仅有一个的如果要从同一目录中的文件夹导入。每后退一步,再加一步。

hi/
  hello.py
how.py

在how.py中

from .hi import hello

如果您想从hello.py导入how

from .. import how