我一直在为 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
关于Python + 子模块 : ImportError: attempted relative import with no known parent package,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/66961262/