翻译:《实用的Python编程》03_04_Modules
目录 | 上一节 (3.3 错误检查) | 下一节 (3.5 主模块)
3.4 模块
本节介绍模块的概念以及如何使用跨多个文件的函数。
模块和导入
任何一个 Python 源文件都是一个模块。
# foo.py
def grok(a):
...
def spam(b):
...
import
语句加载并执行一个模块。
# program.py
import foo
a = foo.grok(2)
b = foo.spam('Hello')
...
命名空间
模块是命名值的集合,有时也称为命名空间。名称是源文件中定义的所有全局变量和函数。导入之后,模块名称用作前缀。因此,称为命名空间。
import foo
a = foo.grok(2)
b = foo.spam('Hello')
...
模块名直接绑定到文件名(foo -> foo.py)。
全局定义
填充模块命名空间的内容是定义在全局(global)作用域中任何内容。考虑定义了相同变量 x 的两个模块。
# foo.py
x = 42
def grok(a):
...
# bar.py
x = 37
def spam(a):
...
在本例中,x
指向不同的变量。一个是 foo.x
,另一个是 bar.x
。不同的模块可以使用相同的名称并且这些名称不会相互冲突。
模块是隔离的。
把模块当做环境
对于所有定义在模块里面的代码而言,模块构成一个封闭的环境。
# foo.py
x = 42
def grok(a):
print(x)
全局变量始终绑定到封闭模块(相同文件),每个源文件都是它自己的小宇宙。
模块执行
导入模块时,模块中的所有语句依次执行(execute),直到到达文件末尾。模块命名空间的内容是所有的全局名称,这些名称在执行过程结束时仍然被定义。如果有脚本语句在全局作用域中执行任务(如打印,创建文件等),您将看到它们在导入模块时运行。
import as
语句
可以在导入模块时更改其名称:
import math as m
def rectangular(r, theta):
x = r * m.cos(theta)
y = r * m.sin(theta)
return x, y
它的作用与普通导入相同,仅仅是重命名模块而已。
from import
语句
from import
语句从模块中选出符号并使它们在局部可访问。
from math import sin, cos
def rectangular(r, theta):
x = r * cos(theta)
y = r * sin(theta)
return x, y
这允许使用模块的某些部分,而不必输入模块前缀。对于经常使用的名称,这非常有用。
导入说明
有关导入的各种变化不改变模块的工作方式。
import math
# vs
import math as m
# vs
from math import cos, sin
...
具体来说,import
始终执行整个文件并且模块仍然是隔离的环境。
import module as
语句只局部地更改名称。在后台,from math import cos, sin
语句仍加载全部的数学模块。当导入完成后,它仅仅将模块中的 cos
和 sin
名称复制到局部命名空间中。
模块加载
每个模块仅加载和执行一次。注意:重复导入仅返回先前所加载模块的引用
sys.modules
是所有已加载模块的字典。
>>> import sys
>>> sys.modules.keys()
['copy_reg', '__main__', 'site', '__builtin__', 'encodings', 'encodings.encodings', 'posixpath', ...]
>>>
注意:当修改模块的源代码后,如果重复import
语句会产生一个常见的困惑。由于模块缓存 sys.modules
,重复导入总是返回之前加载的模块——即使更改已经发生。将修改后的代码加载到 Python 中最安全的方式是退出然后重启解释器。
定位模块
搜索模块时,Python 从路径列表(sys.path)中查询。
>>> import sys
>>> sys.path
[
'',
'/usr/local/lib/python36/python36.zip',
'/usr/local/lib/python36',
...
]
当前工作目录通常是第一个。
模块搜索路径
如前所述,sys.path
包含搜索路径。可以根据需要手动调整 。
import sys
sys.path.append('/project/foo/pyfiles')
也可以通过环境变量添加搜索路径。
% env PYTHONPATH=/project/foo/pyfiles python3
Python 3.6.0 (default, Feb 3 2017, 05:53:21)
[GCC 4.2.1 Compatible Apple LLVM 8.0.0 (clang-800.0.38)]
>>> import sys
>>> sys.path
['','/project/foo/pyfiles', ...]
在大部分情况下,没必要手动调整模块搜索路径。但是,如果尝试导入的 Python 代码位于特殊位置,或者无法从当前工作目录轻松访问,那么就需要手动调整搜索路径了。
练习
因为本练习涉及模块,所以确保在适当的环境中运行 Python 至关重要。模块经常给编程新手带来问题,这些问题与当前工作目录相关或者与 Python 路径设置相关。对于本课程,假定您是在 Work/
目录下编写所有的代码。为了获得最佳结果,应该确保也是在 Work/
目录下运行解释器。否则,需要确保 practical-python/Work
已添加到 sys.path
。
练习 3.11:模块导入
在第 3 节中,我们创建了一个通用目标函数 parse_csv()
用于解析 CSV 数据文件的内容。
现在,我们来看看如何在其它程序中使用该函数。首先,启动一个新的 shell 窗口,进入到放置所有文件的目录中。我们将要导入它们。
启动 Python 交互模式。
bash % python3
Python 3.6.1 (v3.6.1:69c0db5050, Mar 21 2017, 01:21:04)
[GCC 4.2.1 (Apple Inc. build 5666) (dot 3)] on darwin
Type "help", "copyright", "credits" or "license" for more information.
>>>
当Python 交互模式启动后,尝试导入某些之前编写的程序。应该能看到输出和以前一样。强调一下,导入模块会运行模块中的代码。
>>> import bounce
... watch output ...
>>> import mortgage
... watch output ...
>>> import report
... watch output ...
>>>
如果没有代码运行,可能是因为在错误的目录下运行了 Python。现在,尝试导入 fileparse
模块并获取有关该模块的帮助。
>>> import fileparse
>>> help(fileparse)
... look at the output ...
>>> dir(fileparse)
... look at the output ...
>>>
尝试使用 fileparse
模块来读取一些数据:
>>> portfolio = fileparse.parse_csv('Data/portfolio.csv',select=['name','shares','price'], types=[str,int,float])
>>> portfolio
... look at the output ...
>>> pricelist = fileparse.parse_csv('Data/prices.csv',types=[str,float], has_headers=False)
>>> pricelist
... look at the output ...
>>> prices = dict(pricelist)
>>> prices
... look at the output ...
>>> prices['IBM']
106.11
>>>
尝试导入一个函数,以便不用再包含模块名:
>>> from fileparse import parse_csv
>>> portfolio = parse_csv('Data/portfolio.csv', select=['name','shares','price'], types=[str,int,float])
>>> portfolio
... look at the output ...
>>>
练习 3.12:使用库模块
在第 2 节中,编写了 report.py
程序用来生成像下面这样的股票报告:
Name Shares Price Change
---------- ---------- ---------- ----------
AA 100 9.22 -22.98
IBM 50 106.28 15.18
CAT 150 35.46 -47.98
MSFT 200 20.89 -30.34
GE 95 13.48 -26.89
MSFT 50 20.89 -44.21
IBM 100 106.28 35.84
使用该程序并对其进行修改,以便使用 fileparse
模块中的函数完成所有输入文件的处理。为此,将 fileparse
作为模块导入,并修改 read_portfolio()
和 read_prices()
函数以便使用 parse_csv()
函数。
在本练习开始时,请使用交互示例作为指南。之后,应该能够获得与之前完全相同的输出。
练习 3.14:使用更多的库导入
在第 1 节中,编写了一个读取股票投资组合和计算费用的程序 pcost.py
。
>>> import pcost
>>> pcost.portfolio_cost('Data/portfolio.csv')
44671.15
>>>
请修改 pcost.py
文件,以便它能够使用 report.read_portfolio()
函数。
说明
当完成练习后,您应该拥有三个程序。包含通用目的函数 parse_csv()
的fileparse.py
程序。用于生成报告,且包含 read_portfolio()
和 read_prices()
函数的 report.py
程序。最后,利用 report.py
程序中编写的read_portfolio()
函数去计算股票投资组合费用的 pcost.py
程序。
目录 | 上一节 (3.3 错误检查) | 下一节 (3.5 主模块)
注:完整翻译见 https://github.com/codists/practical-python-zh
翻译:《实用的Python编程》03_04_Modules的更多相关文章
- 翻译:《实用的Python编程》InstructorNotes
实用的 Python 编程--讲师说明 作者:戴维·比兹利(David Beazley) 概述 对于如何使用我的课程"实用的 Python 编程"进行教学的问题,本文档提供一些通用 ...
- 翻译:《实用的Python编程》README
欢迎光临 大约 25 年前,当我第一次学习 Python 时,发现 Python 竟然可以被高效地应用到各种混乱的工作项目上,我立即被震惊了.15 年前,我自己也将这种乐趣教授给别人.教学的结果就是本 ...
- 翻译:《实用的Python编程》05_02_Classes_encapsulation
目录 | 上一节 (5.1 再谈字典) | 下一节 (6 生成器) 5.2 类和封装 创建类时,通常会尝试将类的内部细节进行封装.本节介绍 Python 编程中有关封装的习惯用法(包括私有变量和私有属 ...
- 翻译:《实用的Python编程》04_02_Inheritance
目录 | 上一节 (4.1 类) | 下一节 (4.3 特殊方法) 4.2 继承 继承(inheritance)是编写可扩展程序程序的常用手段.本节对继承的思想(idea)进行探讨. 简介 继承用于特 ...
- 翻译:《实用的Python编程》01_02_Hello_world
目录 | 上一节 (1.1 Python) | 下一节 (1.3 数字) 1.2 第一个程序 本节讨论有关如何创建一个程序.运行解释器和调试的基础知识. 运行 Python Python 程序始终在解 ...
- 翻译:《实用的Python编程》03_03_Error_checking
目录 | 上一节 (3.2 深入函数) | 下一节 (3.4 模块) 3.3 错误检查 虽然前面已经介绍了异常,但本节补充一些有关错误检查和异常处理的其它细节. 程序是如何运行失败的 Python 不 ...
- 翻译:《实用的Python编程》03_05_Main_module
目录 | 上一节 (3.4 模块) | 下一节 (3.6 设计讨论) 3.5 主模块 本节介绍主程序(主模块)的概念 主函数 在许多编程语言中,存在一个主函数或者主方法的概念. // c / c++ ...
- 翻译:《实用的Python编程》04_01_Class
目录 | 上一节 (3.6 设计讨论) | 下一节 (4.2 继承) 4.1 类 本节介绍 class 语句以及创建新对象的方式. 面向对象编程(OOP) 面向对象编程是一种将代码组织成对象集合的编程 ...
- 翻译:《实用的Python编程》05_00_Overview
目录 | 上一节 (4 类和对象) | 下一节 (6 生成器) 5. Python 对象的内部工作原理 本节介绍 Python 对象的内部工作原理.来自其它语言的程序员通常会发现 Python 的类概 ...
随机推荐
- BZOJ 4516. [Sdoi2016]生成魔咒【SAM 动态维护不同子串数量】
[Sdoi2016]生成魔咒 动态维护不同子串的数量 想想如果只要查询一次要怎么做,那就是计算各个点的\(len[u]-len[link[u]]\)然后求和即可,现在要求动态更新,我们可以保存一个答案 ...
- Codeforces Round #696 (Div. 2) C. Array Destruction (贪心,multiset)
题意:有\(n\)个数,首先任选一个正整数\(x\),然后在数组中找到两个和为\(x\)的数,然后去掉这两个数,\(x\)更新为两个数中较大的那个.问你最后时候能把所有数都去掉,如果能,输出最初的\( ...
- 【noi 2.2_8758】2的幂次方表示(递归)
题意:将正整数N用2的幂次方表示(彻底分解至2(0),2). 解法:将层次间和每层的操作理清楚,母问题分成子问题就简单了.但说得容易,操作没那么容易,我就打得挺纠结的......下面附上2个代码,都借 ...
- P1091 合唱队形(LIS)
题目描述 NNN位同学站成一排,音乐老师要请其中的(N−KN-KN−K)位同学出列,使得剩下的KKK位同学排成合唱队形. 合唱队形是指这样的一种队形:设K位同学从左到右依次编号为1,2,-,K1,2, ...
- DSC注册Agent失败- InternalServerError
问题 有大概5台Agent Server,注册的时候,发现2台可以成功,其他的不成功. 注册失败的错误日志如下: 初步尝试 首先,Pull Server已经平稳的运行了几年了,此次注册还有部分Agen ...
- LEETCODE - 160【相交链表】
/** * Definition for singly-linked list. * struct ListNode { * int val; * ListNode *next; * ListNode ...
- leetcode11 盛水容器 贪心
这道题,总感觉做过... 先理解题意,何为容器 容器 要求水面高度相同 于是体积就是长方形,高度有两块较高板的低板决定,宽度由两块板间距离决定. 考虑当前最优解,就贪心 从两边开始向内,若能使得体积变 ...
- Leetcode 30 串联所有单词的子串 滑动窗口+map
见注释.滑动窗口还是好用. class Solution { public: vector<int> findSubstring(string s, vector<string> ...
- hdu5693D++游戏 区间DP-暴力递归
主要的收获是..如何优化你递推式里面不必要的决策 之前的代码 这个代码在HDU超时了,这就对了..这个复杂度爆炸.. 但是这个思路非常地耿直..那就是只需要暴力枚举删两个和删三个的情况,于是就非常耿直 ...
- 让你像黑客一样写代码(not really)
让你像黑客一样写代码(not really) http://poznan.tvp.pl 这是一个波兰的视频网站. poznan 波兹南(波兰城市 视屏链接 http://video.sina.com. ...