如何解决Python + 子模块:ImportError:尝试在没有已知父包的情况下进行相对导入 解决方案说明
我一直在为 Python 的相对导入而苦苦挣扎。
我的项目结构如下:
root_dir
├── main.py
├── Pipfile
├── Pipfile.lock
├── unit_tests
│ ├── __init__.py
│ ├── test_first.py
│ └── test_second.py
└── my_package
├── __init__.py
├── first.py
├── second.py
└── third.py
我想将位于 my_package
中的文件中的一组函数导入到 test_first.py
。
根据 official documentation 语法应如下所示:
from ..my_package import first
当我这样做时,我得到以下异常:
Traceback (most recent call last):
File "/home/user/dev/root_dir/my_package/unit_tests/first.py",line 8,in <module>
from ..my_package import first
ImportError: attempted relative import with no known parent package
我还尝试使用以下语法导入文件:
from root_dir.my_package import first
当我这样做时,我得到以下异常:
ModuleNotFoundError: No module named 'root_dir'
值得一提的是,我使用 pipenv
来处理虚拟环境。
知道为什么会这样吗?
谢谢。
解决方法
首先,我将向您提供对我有用的代码。那么,我来做一个简单的说明。这是一个简短的函数,可以让您从根目录导入。
解决方案
import os,sys
def access_root_dir(depth = 1):
current_dir = os.path.dirname(os.path.realpath(__file__))
parent_dir = os.path.dirname(current_dir)
args: list = [parent_dir]
for _ in range(depth):
args.append('..')
rel_path = os.path.join(*args)
sys.path.append(rel_path)
# At the top of test_first.py
access_root_dir(2)
import root_dir
由于根目录是test_first.py
的祖父目录,所以需要进入深度2:
说明
我在我的一个项目中遇到了同样的问题,发现
通过更改 sys.path
修复了从父目录导入的问题。将绝对路径附加到根目录,例如
sys.path.append(abs_path_to_root)
似乎不起作用,除非一切都是包的一部分。默认情况下,Python 不会访问导入中的父目录。
# Case 1: Relative path to parent - works
current_dir = os.path.dirname(os.path.realpath(__file__))
relative_path = os.path.join(current_dir,'..')
sys.path.append(relative_path)
import parent_module_name # This works for me (Python 3.9)
# Case 2: Absolute path to parent - Doesn't work
current_dir = os.path.dirname(os.path.realpath(__file__))
parent_dir = os.path.dirname(current_dir)
sys.path.append(parent_dir)
import parent_module_name # raises ModuleNotFoundError
,
我认为您需要执行“...my_package”。使用三个“...”
".." 表示父 root_dir。最后 ”。”正在访问子文件夹 my_package。
,current working directory is not a package。使用 os.getcwd()
查看它是什么,但很可能是 root_dir
。
这种情况下,可以直接按名称导入顶级包,无需相对导入;您应该直接从 my_package
导入,如下所示:
from my_package import first
...或:
from my_package.first import do_something
复制的文件树:
root_dir
├── main.py
├── my_package
│ ├── __init__.py
│ └── first.py
└── unit_tests
├── __init__.py
└── test_first.py
main.py
的内容:
from unit_tests import test_first
if __name__ == "__main__":
test_first.test_something()
如果您尝试进行相对导入,即:from ..my_package import first
,您将得到:
ImportError: attempted relative import beyond top-level package
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。