python 自己写package 导入 attempted relative import beyond top-level package

一切包的相对导入都是在主程序所在目录之下进行的,不能导入它的上一级目录中的包

自己写好的package可以实现绝对引用相对的引用link1

1.绝对引用。

如下多个package

此时如果想在a.py 文件中调用 package b  或者 c 中的 某个文件或者某个函数,先将 文件夹a 所在工作路径添加到sys.path() 中。

然后再a.py 中  直接 from c.c_fun import add 即可, 跟调用官方package 一样。

2. 相对引用。

在练习Python中package的相对导入时,即

from . import XXX
  或者 

from .. import XXX
  时会遇到这样两个错误:

SystemError: Parent module '' not loaded, cannot perform relative import
  和

ValueError: attempted relative import beyond top-level package

其实这两个错误的原因归根结底是一样的:在涉及到相对导入时,package所对应的文件夹必须正确的被python解释器视作package,而不是普通文件夹。否则由于不被视作package,无法利用package之间的嵌套关系实现python中包的相对导入。

  文件夹被python解释器视作package需要满足两个条件:

  1、文件夹中必须有__init__.py文件,该文件可以为空,但必须存在该文件。

  2、不能作为顶层模块来执行该文件夹中的py文件(即不能作为主函数的入口  模块的__name__ 不能等于__main__)。

  补充:在"from YY import XX"这样的代码中,无论是XX还是YY,只要被python解释器视作package,就会首先调用该package的__init__.py文件。如果都是package,则调用顺序是YY,XX。

       也就是说 你不能在一个x.py 文件中 执行 from .模块名 import *  同时运行 python x.py

  另外,练习中“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:PythonLearningTestImmain.py", line 3, in <module>
from Tom import tom
File "D:PythonLearningTestImKatekate.py", line 4, in <module>
from .. import kate
ValueError: attempted relative import beyond top-level package

可以看到from . import tomBrother 顺利执行,首先执行了Tom文件夹下的__init__.py文件,后来执行了tomBrother.py文件,但是当执行到“from .. import kate”时报错这是因为我们是在TestIm文件夹下把main.py文件作为主函数的入口执行的,因此尽管TestIm文件夹中有__init__.py文件,但是该文件夹不能被python解释器视作package,即Tom package不存在上层packge,自然会报错,相对导入时超出了最高层级的package。

修改方法:

目录树

  test/

  --main.py : from testIm.Tom import tom

  --testIm/

    --__init__.py :print(123)

    --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
123
I'm Tom's __init__!
I'm Tom's Brother!
I'm Kate's __init__!
I'm Tom!
关注公众号 海量干货等你
原文地址:https://www.cnblogs.com/sowhat1412/p/12734226.html