python基础(二十):包

一、包介绍

随着模块数目的增多,把所有模块不加区分地放到一起也是极不合理的,于是Python为我们提供了一种把模块组织到一起的方法,即创建一个包。包就是一个含有__init__.py文件的文件夹,文件夹内可以组织子模块或子包,例如

pool/                #顶级包
├── __init__.py     
├── futures          #子包
│   ├── __init__.py
│   ├── process.py
│   └── thread.py
└── versions.py      #子模块
#1. 在python3中,即使包下没有__init__.py文件,import 包仍然不会报错,而在python2中,包下一定要有该文件,否则import 包报错

#2. 创建包的目的不是为了运行,而是被导入使用,记住,包只是模块的一种形式而已,包的本质就是一种模块

接下来我们就以包pool为例来介绍包的使用,包内各文件内容如下

# process.py
def Process():
	print('this is process')

# thread.py
def Thread():
	print('this is thread')

# versions.py
def check():
    print('check versions’)

# __init__.py文件内容均为空

二、包使用

1、导入包与_init_.py

包属于模块的一种,因而包以及包内的模块均是用来被导入使用的,而绝非被直接执行,首次导入包(如import pool)同样会做三件事:

1.执行包下的__init__.py文件

2.产生一个新的名称空间用于存放__init__.py执行过程中产生的名字

3.在当前执行文件所在的名称空间中得到一个名字pool,该名字指向__init__.py的名称空间,例如pool.xxx和pool.yyy中的xxx和yyy都是来自于pool下的__init__.py,也就是说导入包时并不会导入包下所有的子模块与子包
import pool

pool.versions.check() #抛出异常AttributeError

pool.versions.check()要求pool下有名字versions,进而pool.versions下有名字check。pool.versions下已经有名字check了,所以问题出在pool下没有名字versions,这就需要在pool下的__init__.py中导入模块versions

强调:

  • 关于包相关的导入语句也分为import和from … import …两种,但是无论哪种,无论在什么位置,在导入时都必须遵循一个原则:凡是在导入时带点的,点的左边都必须是一个包,否则非法。可以带有一连串的点,如import 顶级包.子包.子模块,但都必须遵循这个原则。
  • import只适用于导入模块(import 包.子包.模块),from … import …既可以导入模块(from 包 import 模块),又可以导入一个模块文件中的方法(from 包.子包.模块 import check)
  • import导入文件时,产生名称空间中的名字来源于文件,import 包,产生的名称空间的名字同样来源于文件,即包下的__init__.py,导入包本质就是在导入该文件

2、绝对导入与相对导入

针对包内的模块之间互相导入,导入的方式有两种

(1)绝对导入:以顶级包为起始
pool/                #顶级包
├── __init__.py     
├── futures          #子包
│   ├── __init__.py
│   ├── process.py
│   └── thread.py
└── versions.py      #子模块

​重点:包内部的目录结构通常是包的开发者为了方便自己管理和维护代码而创建的,这种目录结构对包的使用者往往是无用的此时通过操作__init__.py可以“隐藏”包内部的目录结构,降低使用难度,比如想要让使用者直接使用pool.check()那么应该:

#pool包下的__init__.py文件:
from pool.versions import check 
from pool.futures.process import Process
from pool.futures.thread import Thread
# 注意这里我原来写的是from pool import versions,只导入了模块,这样是不能实现直接 pool.check(),这样只能pool.versions.check()调用方法。

使用者,只需要向sys.path中加入模块搜索路径即可使用:

# run.py
import sys
sys.path.append(r'/Users/shijiandingyiqingchun/PycharmProjects/untitled/') # pool包就在这个目录下
import pool
pool.check()
pool.process()
pool.thread()
# 至此我们实现了导入pool包可以调用pool包中包括子包中所有的方法或变量名!

有人会问,那子包futures下的__init__.py还有存在的必要吗?

答:有必要,如果我只想导入futures子包中的方法呢?我就可以用import pool.futures,这个时候我们futures下的__init__.py就派上用场了。
(2)相对导入

.代表当前文件所在的目录,…代表当前目录的上一级目录,…上上级目录,以此类推!

#process.py文件:
import ..versions # 导入versions模块
(3)一点建议,两点强调

建议:__init__.py中要导入包中所有模块时,建议使用绝对导入,其他包内的模块之间的导入,建议使用相对导入。

强调:

  • 相对导入只能在包内部使用
  • 无论是import还是from-import,但凡是在导入时带点的,点的左边必须是包,否则语法错误

3、from 包.模块 import *

​ 在使用包时同样支持from pool.futures import * ,毫无疑问代表的是futures下__init__.py中所有的名字,通用是用变量__all__来控制代表的意思

#futures下的__init__.py
__all__=['process','thread']

猜你喜欢

转载自blog.csdn.net/weixin_44571270/article/details/106084841
今日推荐