包(****)
什么是包?
包就是一个包含有__init__.py文件的文件夹
为何要有包?
包的本质是模块的模块的一种形式,包是用来被当做模块导入
创建包的目的不是为了运行,而是被导入使用,记住,包只是模块的一种形式而已,包的本质就是一种模块
如何用包?
import mmm (假设mmm是一个有__init__.py文件的文件夹)
包既然是模块,那去导包的时候肯定也发生着一样的事情
1、产生一个名称空间
2、运行包下的__init__.py文件,将运行过程中产生的名字都丢到1的名称空间中
3、在当前执行文件的名称空间中拿到一个名字mmm,mmm指向1的名称空间
需要注意的:
1. 关于包相关的导入语句也分为import和from ... import ...两种,但是无论哪种,无论在什么位置,
在导入时都必须遵循一个原则:凡是在导入时带点的,点的左边都必须是一个包,否则非法。可以带有一连串的点,
如import 顶级包.子包.子模块,但都必须遵循这个原则。但对于导入后,在使用时就没有这种限制了,
点的左边可以是包,模块,函数,类(它们都可以用点的方式调用自己的属性)。
2、包A和包B下有同名模块也不会冲突,如A.a与B.a来自俩个命名空间
3、import导入文件时,产生的名称空间的名字同样来源于文件,即包下的__init__.py,导入包本质就是在导入该文件
两种导入方式:
绝对导入: 以顶级包为起始
相对导入:.代表当前文件所在的目录,..代表当前目录的上一级目录,依此类推
针对包内部模块之间的相互导入推荐使用相对导入,需要特别强调:
1、相对导入只能在包内部使用,用相对导入不同目录下的模块是非法的
2、无论是import还是from import,但凡是在导入时带点的,点的左边必须是包,否则语法错误
from 包 import *
在使用包时同样支持from pool.futures import * ,毫无疑问*代表的是futures下__init__.py中所有的名字,
通用是用变量__all__来控制*代表的意思
#futures下的__init__.py
__all__=['process','thread']
最后说明一点:包内部的目录结构通常是包的开发者为了方便自己管理和维护代码而创建的,这种目录结构对包的使用者往往是无用的,
此时通过操作__init__.py可以“隐藏”包内部的目录结构,降低使用难度,比如想要让使用者直接使用(不能更改使用者的调用方式)
import pool
pool.check()
pool.ProcessPoolExecutor(3)
pool.ThreadPoolExecutor(3)
需要操作包pool下的__init__.py
from .versions import check
from .futures.process import ProcessPoolExecutor
from .futures.thread import ThreadPoolExecutor
ps:环境变量是以执行文件为准的,所有的被导入的模块或者说后续的其他文件引用的sys.path都是参照执行文件的sys.path