Python 基础笔记(九)

1. 模块化(module)程序设计理念

1.1 模块和包概念的进化史

“量变引起质变”是哲学中一个重要的理论。量变为什么会引起质变呢?本质上理解,

随着数量的增加,管理方式会发生本质的变化;旧的管理方式完全不适合,必须采用新的管

理方式。

程序越来越复杂,语句多了,怎么管理?很自然的,我们会将实现同一个功能的语句封

装到函数中,统一管理和调用,于是函数诞生了。

程序更加复杂,函数和变量多了,怎么管理?同样的思路,“物以类聚”,我们将同一

类型对象的“数据和行为”,也就是“变量和函数”,放到一起统一管理和调用,于是“类

和对象”诞生了。

程序继续复杂,函数和类更加多了,怎么办?好,我们将实现类似功能的函数和类统统

放到一个模块中,于是“模块”诞生了。

程序还要复杂,模块多了,怎么办? 于是,我们将实现类似功能的模块放到一起,于

是“包”就诞生了。

1. Python 程序由模块组成。一个模块对应 python 源文件,一般后缀名是:

.py。

2. 模块由语句组成。运行 Python 程序时,按照模块中语句的顺序依次执行。

3. 语句是 Python 程序的构造单元,用于创建对象、变量赋值、调用函数、控制语句等。

1.2 标准库模块(standard library)

与函数类似,模块也分为标准库模块和用户自定义模块。

Python 标准库提供了操作系统功能、网络通信、文本处理、文件处理、数学运算等基

本的功能。比如:

random(随机数)、math(数学运算)、time(时间处理)、file(文件处理)、

os(和操作系统交互)、sys(和解释器交互)等。

另外,Python 还提供了海量的第三方模块,使用方式和标准库类似。功能覆盖了我们

能想象到的所有领域,比如:科学计算、WEB 开发、大数据、人工智能、图形系统等。

1.3 为什么需要模块化编程

模块(module)对应于 Python 源代码文件(.py 文件)。模块中可以定义变量、函数、

类、普通语句。 这样,我们可以将一个 Python 程序分解成多个模块,便于后期的重

复应用。

模块化编程(Modular Programming)将一个任务分解成多个模块。每个模块就像一个积木一样,便于后期的反复使用、反复搭建

模块化编程有如下几个重要优势:

1. 便于将一个任务分解成多个模块,实现团队协同开发,完成大规模程序

2. 实现代码复用。一个模块实现后,可以被反复调用。

3. 可维护性增强。

1.4 模块化编程的流程

模块化编程的一般流程:

1. 设计 API,进行功能描述。

2. 编码实现 API 中描述的功能。

3. 在模块中编写测试代码,并消除全局代码。

4. 使用私有函数实现不被外部客户端调用的模块函数。

1.5 模块的 API 和功能描述要点

API(Application Programming Interface 应用程序编程接口)是用于描述模

块中提供的函数和类的功能描述和使用方式描述。

模块化编程中,首先设计的就是模块的 API(即要实现的功能描述),然后开始编

码实现 API 中描述的功能。最后,在其他模块中导入本模块进行调用。

我们可以通过help(模块名)查看模块的API。一般使用时先导入模块 然后通过help

1.6 模块的创建和测试代码

每个模块都有一个名称,通过特殊变量__name__可以获取模块的名称。在正常情况

下,模块名字对应源文件名。 仅有一个例外,就是当一个模块被作为程序入口时(主

程序、交互式提示符下),它的__name__的值为“

__main__”。我们可以根据这个特

点,将模块源代码文件中的测试代码进行独立的处理。例如:

import math

math.__name__

#输出'math'

【示例】通过__name==“

__main__”独立处理模块的测试代码

2.1 import 语句导入

import 语句的基本语法格式如下:

import 模块名

#导入一个模块

import 模块 1,模块 2…

#导入多个模块

import 模块名

as 模块别名

#导入模块并使用新名字

import 加载的模块分为四个通用类别:

a.使用 python 编写的代码(

.py 文件);

b.已被编译为共享库或 DLL 的 C 或 C++扩展;

c.包好一组模块的包

d.使用 C 编写并链接到 python 解释器的内置模块;

2.3 import 语句和 from...import 语句的区别

import 导入的是模块。from...import 导入的是模块中的一个函数/一个类。

如果进行类比的话,

import 导入的是“文件”,我们要使用该“文件”下的内容,必

须前面加“文件名称”。from...import 导入的是文件下的“内容”,我们直接使用这

些“内容”即可,前面再也不需要加“文件名称”了。

2.5 模块的加载问题

当导入一个模块时, 模块中的代码都会被执行。不过,如果再次导入这个模块,

则不会再次执行。

Python 的设计者为什么这么设计?因为,导入模块更多的时候需要的是定义模块

中 的 变 量 、 函 数 、 对 象 等 。 这 些 并 不 需 要 反 复 定 义 和 执 行 。 “ 只 导 入 一 次

import-only-once”就成了一种优化。

3.3 导入包操作和本质

上一节中的包结构,我们需要导入 module_AA.py。方式如下:

1. import a.aa.module_AA

在使用时,必须加完整名称来引用,比如:a.aa.module_AA.fun_AA()

2. from a.aa import module_AA

在使用时,直接可以使用模块名。 比如:module_AA.fun_AA()

3. from a.aa.module_AA import fun_AA

直接导入函数

在使用时,直接可以使用函数名。 比如:

fun_AA()

【注】

1. from package import item 这种语法中,

item 可以是包、模块,也可以是函数、

类、变量。

2. import item1.item2 这种语法中,

item 必须是包或模块,不能是其他。

导入包的本质其实是“导入了包的__init__.py”文件。也就是说,”import pack1”意味

着执行了包 pack1 下面的__init__.py 文件。 这样,可以在__init__.py 中批量导入我们需要

的模块,而不再需要一个个导入。

__init__.py 的三个核心作用:

1. 作为包的标识,不能删除。

2. 用来实现模糊导入

3. 导入包实质是执行__init__.py 文件,可以在__init__.py 文件中做这个包的初始化、以及

需要统一执行代码、批量导入。

3.4 用*导入包

import * 这样的语句理论上是希望文件系统找出包中所有的子模块,然后导入它们。

这可能会花长时间等。Python 解决方案是提供一个明确的包索引。

这个索引由 __init__.py 定义 __all__ 变量,该变量为一列表,如上例 a 包下的

__init__.py 中,可定义 __all__ = ["module_A","module_A2"]

这意味着, from sound.effects import * 会从对应的包中导入以上两个子模块;

【注】尽管提供 import * 的方法,仍不建议在生产代码中使用这种写法。

3.5 包内引用

如果是子包内的引用,可以按相对位置引入子模块 以 aa 包下的 module_AA 中导入 a

包下内容为例:

from .. import module_A

#..表示上级目录

.表示同级目录

from . import module_A2

#.表示同级目录

3.6 sys.path 和模块搜索路径

当我们导入某个模块文件时, Python 解释器去哪里找这个文件呢?只有找到这个文

件才能读取、装载运行该模块文件。它一般按照如下路径寻找模块文件(按照顺序寻找,找

到即停不继续往下寻找):

1. 内置模块

2. 当前目录

3. 程序的主目录

4. pythonpath 目录(如果已经设置了 pythonpath 环境变量)

5. 标准链接库目录

6. 第三方库目录(site-packages 目录)

7. .pth 文件的内容(如果存在的话)

8. sys.path.append()临时添加的目录

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值