python中的相对导入//__name__和__package__的作用//以及相对导入的两个经典错误

结论:

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__,所以,当然找不到父包

 

  • 4
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值