我遇到了在__init__.py中导入和在包的模块中使用import as绝对导入的问题。
我的项目有一个子包,在它的__init__.py中,我用from import as语句将一个类从模块“提升”到子包级别。该模块使用绝对导入从该子包导入其他模块。我得到这个错误AttributeError: 'module' object has no attribute 'subpkg'。
示例
结构:pkg/
├── __init__.py
├── subpkg
│ ├── __init__.py
│ ├── one.py
│ └── two_longname.py
└── tst.py
pkg/\u init\uuuu.py为空。
包装/子包装/初始包装:from pkg.subpkg.one import One
包装/子包装/一个.py:import pkg.subpkg.two_longname as two
class One(two.Two):
pass
包装/子包装/two_longname.py:class Two:
pass
包装/tst.py:from pkg.subpkg import One
print(One)
输出:$ python3.4 -m pkg.tst
Traceback (most recent call last):
File "/usr/lib/python3.4/runpy.py", line 170, in _run_module_as_main
"__main__", mod_spec)
File "/usr/lib/python3.4/runpy.py", line 85, in _run_code
exec(code, run_globals)
File "/home/and/dev/test/python/imptest2/pkg/tst.py", line 1, in
from pkg.subpkg import One
File "/home/and/dev/test/python/imptest2/pkg/subpkg/__init__.py", line 1, in
from pkg.subpkg.one import One
File "/home/and/dev/test/python/imptest2/pkg/subpkg/one.py", line 1, in
import pkg.subpkg.two_longname as two
AttributeError: 'module' object has no attribute 'subpkg'
工作区
有一些改变使它起作用:清空pkg/subpkg/__init__.py并直接从pkg.subpkg.one导入。
我不认为这是一个选项,因为AFAIK“提升”到包级别是可以的。这是来自an article的引用:One common thing to do in your __init__.py is to import selected
Classes, functions, etc into the package level so they can be
conveniently imported from the package.
在one.py中将import as更改为from import:from pkg.subpkg import two_longname
class One(two_longname.Two):
pass
这里唯一的缺点是我不能为模块创建短别名。我是从“begueraj”的回答中得到这个想法的。
也可以使用one.py中的相对导入来解决问题。但我认为这只是解决办法的一个变种。
问题有人能解释一下这里到底发生了什么吗?为什么__init__.py中的导入和import as的使用会导致这样的问题?
有更好的解决办法吗?
原始示例
这是我最初的例子。这不太现实,但我不会删除它,所以@begueraj的回答仍然有意义。
pkg/\u init\uuuu.py为空。
包装/子包装/初始包装:from pkg.subpkg.one import ONE
包装/子包装/一个.py:import pkg.subpkg.two
ONE = pkg.subpkg.two.TWO
包装/子包装/2.py:TWO = 2
包装/tst.py:from pkg.subpkg import ONE
输出:$ python3.4 -m pkg.tst
Traceback (most recent call last):
File "/usr/lib/python3.4/runpy.py", line 170, in _run_module_as_main
"__main__", mod_spec)
File "/usr/lib/python3.4/runpy.py", line 85, in _run_code
exec(code, run_globals)
File "/home/and/dev/test/python/imptest/pkg/tst.py", line 1, in
from pkg.subpkg import ONE
File "/home/and/dev/test/python/imptest/pkg/subpkg/__init__.py", line 2, in
from pkg.subpkg.one import ONE
File "/home/and/dev/test/python/imptest/pkg/subpkg/one.py", line 6, in
ONE = pkg.subpkg.two.TWO
AttributeError: 'module' object has no attribute 'subpkg'
最初我在one.py中有这个:import pkg.subpkg.two as two
ONE = two.TWO
在这种情况下,导入时会出现错误(就像在我的原始项目中也使用import as)。