结论:
1:相对导入只能采用from . import 的形式,import不支持相对导入
2:执行文件所在的文件夹内的各个模块之间(.py文件)不可以使用相对导入,这一点在后面说明
3:相对导入只能在包内执行,且只能导入包内模块
第一点不证明
证第二点和第三点:
在证明之前,需要先讲两个内置属性__name__和__package__。
1:__name__:用于标记文件的名称,这里有3点需要注意
①:执行模块的__name__ == __main__
②:与执行模块在同一个文件夹下面的模块,其__name__就是模块名,在执行模块所在文件夹的子 文件夹下的模块名称包含该子文件夹路径
③:与执行模块不在同一个文件夹下的模块的__name__包含有该文件所在的文件夹的路径
下面给出实验结果
第一个,执行文件的__name__ == __main__(此处直接执行A_1模块)
第二个:与执行模块在同一个文件夹下面的模块,其__name__ == 模块名(此处在A_1模块下导入A模块)
而在执行模块所在文件夹的子文件夹下的模块__name__包含子文件夹路径:
第三个:与执行模块不在同一个文件夹下的模块,其__name__包含其文件夹路径(这里在B中导入包xjm,在包xjm的__init__.py文件中导入A文件),可以发现,此时A模块的__name__中包含有前缀xjm这个包名:
产生这样差异的原因在于:
python导入模块的时候是先搜索sys.modules,再搜索sys.path,这一块不清楚的同学可以参考我上一篇文章:python的import机制
一个工程默认的搜索路径为工程根目录和执行模块所在目录,所以与执行模块在同一个包下面的模块,其__name__不需要包含包的名字,而在其他包下面的模块的名字包含其包的名字。
2:__package__:用于标记文件所在的文件夹路径(包路径)
①:执行模块的__package__为None。
②:与执行模块在同一个包下面的模块没有__package__(至少print不出来)
③:其他模块的__package__为模块__name__的前缀
证明:
第一个:执行模块的__package__ == None(此处直接执行A_1模块)
第二个:与执行模块在同一个包下面的模块没有__package__(至少print不出来)(在A_1中导入A)。
结合上面的__name__可以很好的解释,因为这个时候A的__name__ == 'A',而__package__实际上就是__name__的模块名的前缀,所以'A'没有前缀,当然也就没有__package__(至少print不出来)
第三点:不和执行模块在同一个包下面的模块的__package__是其模块__name__中的包的路径(在B中导入包xjm,在包xjm的__init__.py文件中导入A模块)
现在,进入我们的主题相对导入
问什么说相对导入只能是在包内相互导入,且执行模块所在包的模块不可以使用相对导入,因为相对导入的搜索路径是按照__package__来的。
于是现在我们可以理解相对导入常见的两个错误了:
1:第一个错误:ValueError: attempted relative import beyond top-level package
我们在B中导入包xjm,在包的__init__.py文件中导入A_2模块,在A_2模块中打印其__name__和__package__,并导入包ces中的模块dasdas,出现如下结果:
可以看到此时A_2中的__package__为xjm,当使用from ..的时候,已经超出了__package__的顶层包xjm了,所以会报错。
2:第二个错误:ImportError: attempted relative import with no known parent package
原因就在于,执行模块及其所在包的模块没有__package__,所以,当然找不到父包