你常常看到的 __init__.py 到底是个啥?

共 1335字,需浏览 3分钟

 ·

2020-04-15 23:27


当你去看一些 Python 相关的项目时,常常会看到 __init__.py,当你使用某些编辑器创建 Python Package 的时候,它也会自动给你生成一个 __init__.py 文件



22ac9282e1a122a065b9234e9ae1499d.webp



这玩意有什么用呢?


我们知道, Python 中的包是可以包含多个 py 模块的,我们可以在不同的地方通过包名区分使用这些模块。
话不多说,咱们先来创建一下这样的目录:


97eabd9a220c173482fd483a77b60b4b.webp


我们在这里面创建了三个 Python 子包,里面都有 __init__ 和 module 的 py 文件。


我们分别在不同包下的 __init__ 中写一个 print 语句:




506fee8917200651028261846b240986.webp



接着我们进入 Python ,分别来导入这些模块:




5522e9333124bc9c73f31060717ed5ff.webp


可以看到,当我们导入父模块中的子模块的时候,它会优先执行父模块中的 init ,接着会执行指定模块中的 init。
当然,只是导入父模块的时候只会执行父模块中的 init:


13649b604e1cd7f4aa57402886d2a4bf.webp


也就是说,当我们去 import 一个 Package 的时候,它会隐性的去执行 __init__.py , 而在 __init__.py 中定义的对象,会被绑定到当前的命名空间里面来。
比如有时候我们会这样去导入一个包下的所有模块,会这样操作:



bb388cc73cec7e20ab386e9d17e3f358.webp


但这个时候你会发现并没有将相关的子模块导入进来:


3f15058d9c7d1c8b4678dda6f677e319.webp


这时候你可能想到了,可以在父模块中的 __init__.py 做文章,先把它们导入进来不就行了:



c77dbda5216b917d84c11c2f9486ea1d.webp


这里的 __all__ 相当于导入 [] 里面定义的模块。

这次再导入:



0b8815a519c4380249c322c7c37de733.webp


可以看到,所有子模块就都一并导入进来了。
当然,你也可以在 __init__.py 做一些初始化的操作,比如数据库 session 的创建:


c4c7e3b8f6805df87b7cce866afc53f4.webp


其实在 Python3.2 版本之前,定义的 Package 下面一定要有 __init__.py 文件,这样 Python 才知道它是一个 Package,才可以寻找到相关模块的路径从而被 import。
而在 Python3.2 之后的版本就不需要再额外的去专门创建一个 __init__.py 来告诉 Python 它是一个 Package 了,因为现在创建的包叫 Namespace package, Python 可以自动搜寻 Package 路径,哪怕你的父包路径发生了改变,你在下次导入的时候, Python 还是会自动重新搜索包路径。
我们把刚刚定义的 __init__ 都给删掉试试:



e0f6acc71c379763ebb625ec8b5842d6.webp


接着我们在 Python3.8 版本导入看看:



017930381c0f1aa2317f72667fb9a3d8.webp


可以看到,尽管我们在 package 中没有定义 __init__.py,依然可以导入使用。

以同样的形式,我们在 Python2 中导入看看:



02e4e121dea790f7027b9bd83f31d64d.webp


可以看到,它被整懵逼了...
综上,__init__.py 会在 import 的时候被执行,而空的 __init__.py 在 Python 新版本中已经不需要你额外去定义了,因为就算你不定义 init, Python 也知道你导入的包路径,但是如果你想要做一些初始化操作,或者像我们刚刚说的预先导入相关的模块,那么定义 __init__.py 还是很有必要的哟。
希望对你有帮助,那么我们下回见,peace!


点击留言

a64bcbe96c540d141c06130bb9c40ce2.webp扫一扫学习 Python 没烦恼




相关
92dee96a716f119a921ba3d11acd790b.webp

你常常在写的 if __name__ == '__main__'  到底是个啥?

浏览 34
点赞
评论
收藏
分享

手机扫一扫分享

分享
举报
评论
图片
表情
推荐
点赞
评论
收藏
分享

手机扫一扫分享

分享
举报