
当一个模块(如a.py)需要导入同目录下的另一个模块(如b.py)时,必须使用相对导入语法(如from .b import b),否则会因python解释器无法解析模块路径而报modulenotfounderror。
在Python包结构中,模块间的导入行为严格依赖于执行上下文和导入语法类型。你遇到的 ModuleNotFoundError: No module named 'B' 错误,根本原因在于:A.py 中写的 from B import B 是绝对导入,Python 会在 sys.path 的顶层(如当前工作目录、site-packages等)搜索名为 B 的模块,而不会自动查找 A.py 所在的同一包目录 —— 即使 B.py 就在旁边。
✅ 正确做法是:在 A.py 中改用显式相对导入:
# dir/A.py
from .B import B # ✅ 正确:. 表示当前包(即 dir/)
class A:
def __init__(self):
self.b = B() # 示例使用同时确保以下前提条件全部满足:
- dir/ 下存在 __init__.py(可以为空),使其被识别为合法的 Python 包;
- 导入操作(如 from dir.A import A)不能在脚本顶层直接以 python C.py 方式运行,除非 C.py 不是作为主模块(main)而是作为模块被导入;更稳妥的方式是:
- 将 C.py 移至项目根目录外,或
- 使用 -m 参数运行(推荐):
python -m parent_dir.C # 前提:parent_dir 是包,且有 __init__.py
- 或者,在 C.py 中通过 if __name__ == "__main__": 安全调用,并确保启动时工作目录为 parent_dir。
⚠️ 注意事项:
立即学习“Python免费学习笔记(深入)”;
- ❌ 不要写 import B 或 from B import * —— 这仍是绝对导入,无效;
- ❌ 不要在 A.py 中写 from dir.B import B —— 这是跨包引用,在 A.py 内部属于冗余且易错(dir 对 A.py 而言并非父级包名);
- ✅ 相对导入仅在包内模块间有效,且必须以 .(当前包)、..(父包)开头;
- ? 若未来需支持 python C.py 直接运行,可在 C.py 开头动态修正路径(不推荐用于生产):
# C.py(不推荐,仅作兼容临时方案) import sys from pathlib import Path sys.path.insert(0, str(Path(__file__).parent / "dir")) from A import A # 此时 A.py 需仍用 from .B import B
总结:模块协作的核心原则是「明确作用域,尊重包结构」。只要 A.py 和 B.py 同属一个包(由 __init__.py 标识),就应统一使用相对导入声明依赖,这是 PEP 328 规定的标准实践,也是可维护、可移植的唯一可靠方式。










