在练习Python中package的相对导入时,即python
from . import XXX
或者 linux
from .. import XXX
时会遇到这样两个错误:shell
SystemError: Parent module '' not loaded, cannot perform relative import
和函数
ValueError: attempted relative import beyond top-level packageorm
其实这两个错误的缘由归根结底是同样的:在涉及到相对导入时,package所对应的文件夹必须正确的被python解释器视做package,而不是普通文件夹。不然因为不被视做package,没法利用package之间的嵌套关系实现python中包的相对导入。it
文件夹被python解释器视做package须要知足两个条件:ast
一、文件夹中必须有__init__.py文件,该文件能够为空,但必须存在该文件。form
二、不能做为顶层模块来执行该文件夹中的py文件(即不能做为主函数的入口)。test
补充:在"from YY import XX"这样的代码中,不管是XX仍是YY,只要被python解释器视做package,就会首先调用该package的__init__.py文件。若是都是package,则调用顺序是YY,XX。import
另外,练习中“from . import XXX”和“from .. import XXX”中的'.'和'..',能够等同于linux里的shell中'.'和'..'的做用,表示当前工做目录的package和上一级的package。
举个例子:
目录树
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, in
from cat import cat
File "D:\PythonLearning\TestIm2\cat\cat.py", line 4, in
from .. import dog
ValueError: attempted relative import beyond top-level package
复制代码
能够看到from . import tomBrother顺利执行,首先执行了Tom文件夹下的__init__.py文件,后来执行了tomBrother.py文件,可是当执行到“from .. import dog”时报错,这是由于咱们是在TestIm文件夹下把main.py文件做为主函数的入口执行的,所以尽管TestIm文件夹中有__init__.py文件,可是该文件夹不能被python解释器视做package,即Tom package不存在上层packge,天然会报错,相对导入时超出了最高层级的package。
修改方法:
目录树
test/
--main.py : from testIm.Tom import tom
--testIm/
--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
运行文件:main.py
结果:
I'm top's init!
I'm Tom's init!
I'm Tom's Brother!!
I'm Kate's init! I'm Tom! 即主函数入口不在TestIm中,则TestIm和其一样包含__init__.py文件的子文件夹都被python解释器视做package,造成相应的嵌套关系。能够正常使用from . import XXX和from .. import XXX。