python包与模块
Python基础-包与模块
摘要
- 为重用以及更好的维护代码,
Python使用了模块与包;一个Python文件就是一个模块,包是组织模块的特殊目录(包含__init__.py文件)。 - 模块搜索路径,
Python解释器在特定的目录中搜索模块,运行时sys.path即搜索路径。 - 使用
import关键字导入模块,注意import *与__all__的关系。
1. 模块与导入
A module is a file containing Python definitions and statements
Python模块就是包含定义以及语句的文件,文件名是模块的名字加上.py后缀。
1.1 为重用而生
假设有一个完成特定功能,很好用的函数或者类。为了使用这个功能,不得不把这段代码复制到需要使用的每一个文件中。重复代码是编程的大忌,如果功能实现需要修改,会不得不修改每一个出现的地方,这是反人类的。
重用能够很好的解决这一问题,实际上,函数,类等结构在一定程度上也为重用提供了便利。
Python中,将一系列相关的函数,类等组织在一个文件中,每一个文件都是一个Python模块。
1.2 导入模块
使用import关键字导入模块(模块需在搜索路径中):
- import sys;基础导入语句。
- import sys as system;为导入的名字起别名。
- from sys import path;导入模块特定元素。
- from sys import *;从sys中导入全部可导入名字
import-only-once
模块只导入一次这种行为在大多数情况下是一种实质性的优化,在同一个解释器生命周期内,多次使用import语句导入同一个模块,导入只发生一次。
这一点可以在模块中加入输出语句证明。
import *与__all__
使用import *可能会污染当前模块的名字空间,导入了一些不需要引用的名字。因此不推荐使用。
事实上,规范的第三方模块会提供一个模块公共接口,暴露该模块可用的接口。公共接口由模块名为__all__的列表定义。
如定义名为mtest1的模块:
__all__ = ['test1', 'test12']
def test1():
print('test1')
def test11():
print('test11')
def test12():
print('test12')
使用全部导入的方式:
>>> form mtest1 import *
>>> dir()
>>> ['__annotations__', '__builtins__', '__doc__', '__loader__','__name__', '__package__', '__spec__', 'test1', 'test12']
可以看到函数test11()并没有被导入,这就是__all__的作用了。
2. 包与其构建
为了更好组织模块,将模块分组为包(package)。
2.1 包是特殊模块
从文件系统上看,包就是模块所在目录。为使Python解释器将其区别普通目录作为包看待,包中必须直接包含一个名为__init__.py的文件(模块)。
包基本上就是另外一类模块,不同的地方在于包能包含其他模块与包。包作为一个模块,其内容其实就是文件__init__.py(模块)的内容。
如名为constants的包,文件constants/__init__.py如下:
PI = 3.14
那么可以将包constants作为普通模块对待:
import constants
print(constants.PI)
2.2 构建包
如果要构建一个名为drawing的包,其中包含shapes和colors模块,需要创建一下目录和文件:
| 文件/目录 | 描述 |
|---|---|
| ~/python | 加入到搜索路径中的目录 |
| ~/python/drawing | 包目录(drawing包) |
| ~/python/drawing/__init__.py | 包代码(drawing模块) |
| ~/python/drawing/colors.py | color模块 |
| ~/python/drawing/shapes.py | shapes模块 |
假设已经将~/python作为搜索目录。依照这个设置,下列导入语句都是合法的:
import drawing# 导入drawing包(即__init__.py模块)import drawing.colors# 导入colors模块,使用drawing.colors.attr的方式引用from drawing import shapes# 导入shapes模块
__all__变量
与模块的__all__变量相似,包的__all__变量决定了使用from package import *导入的子模块。
如以上drawing包的__init__.py文件内容如下:
__all__ = ['colors']
那么使用from drawing import *只会导入colors模块。
3. 搜索路径
现在已经编写完了一个很好用的模块,并且通过了测试。那么如何让这个模块可用呢?即如何让这个模块具备可导入到其他模块的能力。
3.1 搜索模块
当使用import语句导入模块时,Python解释器通过以下方式搜索模块:
- 首先搜索
built-in模块 - 最后搜索变量
sys.path提供的路径列表
sys.path在解释器启动时从以下位置初始化:
- 当前脚本路径
- 环境变量
PYTHONPATH指定的路径集合 - 安装默认路径
sys.path初始化完成后,可以在运行时修改。
3.2 让模块可用
那么现在若要使模块可用,一是将其放置到已有的搜索路径下,二是指定模块所在路径为搜索路径。
一般情况下,若选择第一种方式,我们将模块放置到Python安装路径的\lib\site-packages下,这个目录是专门用来安装第三方模块的。正如该目录下的README文件展示的那样:
This directory exists so that 3rd party packages can be installed here. Read the source for site.py for more details.
若选择第二种方式,直接将模块所在目录加入到环境变量PYTHONPATH中即可。
值得注意的是,可以在\lib\site-packages路径下新建一个名为user_lib.pth的文件,内容是需要搜索的路径,一行一个,也可以将指定路径加入到搜索目录中:

python包与模块的更多相关文章
- python 包和模块
一. 模块 使用内置函数vars()可以查看当前环境下有哪些对象(变量.函数.类) from 模块 import *: 不会导入以下划线开头的对象 只会导入__all__中定义了的对象(__all__ ...
- Python 包、模块、函数、变量作用域
Python 项目的组织结构 - 包 -- 模块 --- 类 ---- 函数.变量 Python是利用包和模块来组织一个项目的. 包: 包的物理表现是一个文件夹,但是一个文件夹却不一定是个包, ...
- 关于python包,模块,.pyc文件和文件导入理解
参考文献 一.包 包是一个文件夹,用来存放模块和子包. 包里一般会有一个__init__.py的文件(也可以没有). 包里会有一个__pycache__文件夹,存放.py文件经解释器解释后的中间字节码 ...
- python包-logging-hashlib-openpyxl模块-深浅拷贝-04
包 包: # 包是一系列模块文件的结合体,表现形式是文件夹,该文件夹内部通常会包含一个__init__.py文件,本质上还是一个模块 包呢,就是前两篇博客中提到的,模块的四种表现形式中的第三种 # 把 ...
- python 包和模块间的引入
##############################总结####################### 主要内容: 1. 模块 2. import 3. from xxx import xxx ...
- python包与模块导入
一 .module 通常模块为一个文件,直接使用import来导入就好了.可以作为module的文件类型有".py".".pyo".".pyc&quo ...
- Pycharm 导入 Python 包、模块
1.点击File->settings 2.选择Project Interpreter,点击右边绿色的加号添加包 3.输入你想添加的包名,点击Install Package 4.可以在Pychar ...
- Python(五) 包、模块、函数与变量作用域
一.while循环与使用场景 CONDITION=1 while CONDITION <=5 : CONDITION +=1 print("hello") else: pri ...
- 17.Python略有小成(包,logging模块)
Python(包,logging模块) 一.包 什么是包 官网解释 : 包是一种通过使用'.模块名'来组织python模块名称空间的方式 , 具体来讲 , 包就是一个包含有__ init __.py文 ...
随机推荐
- 《Java设计模式》之解释器模式
解释器模式是类的行为模式.给定一个语言之后,解释器模式能够定义出其文法的一种表示,并同一时候提供一个解释器. client能够使用这个解释器来解释这个语言中的句子. 解释器模式的结构 以下就以一个示意 ...
- IPV6相关RFC文档
1. 通用 IPv6的通用RFC和Internet草案 RFC# 类 标题 1752 标准记录 对IP下一代协议的建议 1924 资料 IPv6地址的压缩表示法 2851 标准记录 Internet网 ...
- java学习笔记:文件名区分大小写
我按照网上的教程,写了JAVA第一个程序:Hello World!,出了两个问题,都栽在 大小写 上. public class Hello { public static void main(Str ...
- 【POI 2010】 Antisymmetry
[题目链接] https://www.lydsy.com/JudgeOnline/problem.php?id=2084 [算法] manacher [代码] #include<bits/std ...
- [模板] manacher(教程)
还是不会马拉车啊.今天又学了一遍,在这里讲一下. 其实就是一个很妙的思路,就是设置一个辅助的数组len,记录每个点的最大对称长度,然后再存一个mx记录最大的对称子串的右端点.先开二倍数组,然后一点点扩 ...
- MySQL 1071错误解决办法
今天在使用mysql时,又遇到了如博文标题所示的问题,以前针对该问题未进行记录,今天特意进行说明存档. 该问题是由键值字段长度过长导致.mysql支持数据库表单一键值的最大长度不能超过 ...
- Three Kingdoms(优先队列+bfs)
http://acm.hdu.edu.cn/showproblem.php?pid=3442 题意:ABCD有各自的攻击力与攻击范围,刘备只能走"C"与".", ...
- codevs1154能量项链(环形dp,区间dp)
1154 能量项链 2006年NOIP全国联赛提高组 时间限制: 1 s 空间限制: 128000 KB 题目等级 : 黄金 Gold 题目描述 Description 在Mars星球上 ...
- 数值分析常见算法C++实现
1.1-有效数字丢失现象观察 #include<bits./stdc++.h> using namespace std; double f1(double x) { )-sqrt(x)); ...
- JavaScript学习五
2019-06-02 09:53:42