python 2.5中的相对导入

2024-09-25 04:21:31 发布

您现在位置:Python中文网/ 问答频道 /正文

我知道Python中同样的导入问题有很多问题,但似乎没有人能够提供正确使用的清晰示例

假设我们有一个包mypackage,包含两个模块foobar。在foo内部,我们需要能够访问bar

因为我们还在开发它,mypackage不在sys.path

我们希望能够:

  • 导入mypackage.foo
  • foo.py作为脚本运行,并从__main__部分执行示例用法或测试
  • 使用Python 2.5

我们必须如何在foo.py中进行导入,以确保它在所有这些情况下都能工作

# mypackage/__init__.py
...

# mypackage/foo/__init__.py
...

# mypackage/bar.py  
def doBar()
    print("doBar")

# mypackage/foo/foo.py
import bar # fails with module not found
import .bar #fails due to ValueError: Attempted relative import in non-package

def doFoo():
    print(doBar())

if __name__ == '__main__':
    doFoo()

Tags: 模块pyimport示例fooinitmaindef
2条回答

请查看PEP 328中的以下信息:

Relative imports use a module's __name__ attribute to determine that module's position in the package hierarchy. If the module's name does not contain any package information (e.g. it is set to '__main__') then relative imports are resolved as if the module were a top level module, regardless of where the module is actually located on the file system.

当您将foo.py作为脚本运行时,该模块的__name__'__main__',因此您无法进行相对导入。即使mypackagesys.path上,这也是正确的。基本上,如果模块已导入,则只能从该模块进行相对导入

以下是解决此问题的几个选项:

1)在foo.py中,检查__name__ == '__main__',并有条件地将mypackage添加到sys.path

if __name__ == '__main__':
    import os, sys
    # get an absolute path to the directory that contains mypackage
    foo_dir = os.path.dirname(os.path.join(os.getcwd(), __file__))
    sys.path.append(os.path.normpath(os.path.join(foo_dir, '..', '..')))
    from mypackage import bar
else:
    from .. import bar

2)始终使用from mypackage import bar导入bar,并以mypackage自动可见的方式执行foo.py

$ cd <path containing mypackage>
$ python -m mypackage.foo.foo

我的解决方案看起来更干净,可以与所有其他导入一起位于顶部:

try:
   from foo import FooClass
except ModuleNotFoundError:
   from .foo import FooClass

相关问题 更多 >