包的定义:具有 __init__.py
文件的文件夹就是一个包。
包的目的是用来管理模块。
包的本质就是一个模块。模块可以导入,包也可以导入。
在 Python 2 中,使用 import 导入一个包,如果包中没有 __init__.py
文件,会报错;
在 Python 3 中,使用 import 导入一个包,如果包中没有 __init__.py
文件,不会报错,但也无法正常使用。
在导入时只要使用点操作,就是在导入包。导入时,.
前面的必须是一个包。
我们在项目的根目录 PyProject
下建立一个名为 bake
的包,然后按照下面的目录树创建包和 py
文件:
1 | ----bake\ |
创建完成后是这个样子的。需要注意的是,bake
文件夹一定要直接放在项目的根目录 PyProject
下,否则接下来的导入过程可能会出问题。
在文件 policy.py
中写入代码:
1 | name = 'alex' |
在文件 versions.py
中写入代码:
1 | name = 'meet' |
在文件 manage.py
中写入代码:
1 | neme = 'taibai' |
也就是这个样子:
导入包可以使用相对路径和绝对路径两种方法:
绝对路径:bake1.api.policy
从最外层的开始查找
相对路径:取决于在哪个文件中启动,.
代表当前目录
导入包的方式最基本的结构为:
1 | import 包.包.模块 |
比如,我们想在 test.py
文件中导入 version 模块,并使用其中的 name
变量:
1 | import bake.api.versions |
输出的结果为:
1 | api_versions |
我们成功导入了包,也成功调用了包中的变量。但是每次使用的时候都要使用长长的绝对路径,十分麻烦。这时,我们可以给导入的包取一个别名,来方便调用:
1 | import bake.api.policy as p |
输出的结果为:
1 | api_policy |
我们也可以通过 from 方法来导入包中指定的函数或变量,而不是将包整体导入。其基本结构为:
1 | from 包.包.模块 import 函数,变量,* |
比如,我们要导入模块 manage
中的变量 name
,就可以这样:
1 | from bake.cmd.manage import name |
输出的结果为:
1 | cmd_manage |
我们说过,包的本质也是模块,但是如果我们直接导入包,却发现无法直接调用其中的模块中的变量:
1 | from bake import api |
程序报错,报错信息为:
1 | Traceback (most recent call last): |
这是因为我们没有指明 api 包所需要导入的文件。这时,我们就要在 api 包下的 __init__.py
文件中指明导入的文件:
1 | from .policy import * |
这样,程序就可以顺利运行啦。
如果我们直接在 test 文件中导入 bake 中的所有模块中的函数和变量,就可以这样:
1 | from bake import * |
输出的结果为:
1 | api_policy |
因为变量名都命名为 name
,相互覆盖之后,只剩下了 taibai。
总结一下,当程序运行到 from bake import *
命令时,一共经历两个步骤:
- 先在 bake 的
__init__.py
中导入 bake 下的所有的包 - 在 bake 下的所有包中的
__init__.py
中导入每个包下的模块
模块与包的关系为:
- 模块 - 软件开发规范 - 包
- 按照软件开发规范创建包,包中存放模块。