参考来源:https://www.cnblogs.com/ArsenalfanInECNU/p/5346751.html
在导入python的package时,使用 . 或者 .. 来代表当前文件夹和上层文件夹的形式为相对导入。例如,【from . import foo】【from .. import foo】
在执行python的相对导入时,可能会抛出如下的错误:
【SystemError: Parent module''notloaded, cannot perform relativeimport】
和
【ValueError: attempted relativeimportbeyond top-level package】
其实这两个错误的原因归根结底是一样的:在涉及到相对导入时,package所对应的文件夹必须正确的被python解释器视作package,而不是普通文件夹。否则由于不被视作package,无法利用package之间的嵌套关系实现python中包的相对导入。
文件夹被python解释器视作package需要满足两个条件:
1、文件夹中必须有__init__.py文件,该文件可以为空,但必须存在该文件。
2、不能作为顶层模块来执行该文件夹中的py文件(即不能作为主函数的入口)。
例:有如下的文件目录树:
testIm/
--__init__.py
--main.py : from Tom import tom
--Tom/
--__init__.py : print("I'm Tom's __init__!")
--tom.py : from . import tomBrother, from .. import Kate,print("I'm Tom!")
--tomBrother.py print(I'm Tom's Brother!)
--Kate/
--__init__.py : print("I'm Kate's __init__!")
--kate.py
运行文件:main.py
结果:
I'm Tom's__init__!
I'm Tom's Brother!
Traceback (most recent call last):
File "D:\PythonLearning\TestIm2\main.py", line 3,infromcatimport cat
File "D:\PythonLearning\TestIm2\cat\cat.py", line 4,infrom..import dog
ValueError: attempted relative importbeyond top-level package
>>>
可以看到
from . import tomBrother 是可以被执行的,而from .. import Kate 会报错,这是因为我们是在TestIm文件夹下把main.py文件作为主函数的入口执行的,因此尽管TestIm文件夹中有__init__.py文件,但是该文件夹不能被python解释器视作package,即Tom package不存在上层packge,自然会报错,相对导入时超出了最高层级的package。
解决办法:新建文件夹testImPack, 将除了main.py 的其他模块和包全部移入testImPack中。
修改后的目录树如下:
testIm/
--main.py : from testIm.Tom import tom
--testImPack/
--__init__.py
--Tom/
--__init__.py : print("I'm Tom's __init__!")
--tom.py : from . import tomBrother, from .. import Kate,print("I'm Tom!")
--tomBrother.py print(I'm Tom's Brother!)
--Kate/
--__init__.py : print("I'm Kate's __init__!")
--kate.py
补充:在"from YY import XX"这样的代码中,无论是XX还是YY,只要被python解释器视作package,就会首先调用该package的__init__.py文件。如果都是package,则调用顺序是YY,XX。