网站首页 > 技术文章 正文
Scalers点评:在2015年,ScalersTalk成长会完成Python小组完成了《Python核心编程》第1轮的学习。到2016年,我们开始第二轮的学习,并且将重点放在章节的习题上。Python小组是成长会内部小组,如果你想和我们一起学习Python,你需要是成长会成员,并且完成相关进群任务。
我们的节奏是行动是一周一章学到位,章节习题都会一个一个过。通过到位地执行,达到实质性的提升。
往期日志:
本周学习情况
本周(20160321-20160327)学习第10章,章节内容为《错误和异常》。本周复盘主持人为Seven。
本章主要内容
本章主要介绍了什么是异常,如何生成异常,异常处理,Python对异常的支持,如何创建自定义的异常类、断言。
本章主要知识点
10.1 什么是异常
10.1.1 错误
当 Python 检测到一个错误时,解释器会指出当前语句已经无法继续执行下去, 这时候就出现了异常。
语法错误:必须在编译前解决。
逻辑错误:运行时出现,如:非法的输入、边界值溢出等。
10.1.2 异常
异常是程序出现错误而在正常控制流以外采取的行为。
该行为分为两个阶段:
产生异常:当符合异常的产生条件时,解释器就会触发一个异常。异常可以是Python解释器触发的,也可以是coder手动触发的。
处理异常:异常引发后,指示程序如何执行,是忽略错误继续执行还是终止程序。
异常处理的好处:使得程序具有在运行时管理错误的能力,在错误发生时采取可靠的补救措施。
10.2 Python 中的常见异常
ZeroDivisionError
:除数为零NameError
:访问没有定义的变量SyntaxError
:唯一不是在运行时发生的异常,语法错误导致程序编译失败。IndexError
:请求的索引超出序列范围KeyError
:请求一个不存在的字典关键字IOError
:输入/输出错误AttributeError
:访问未知的对象属性
10.3 检测和处理异常
检测和处理异常有两种主要形式: try-except
和try-finally
。
10.3.1 try-except
语句
try: try_suite # watch for exceptions here except Exception[, reason]: except_suite # exception-handling code
try-except
的处理流程:
异常被引发后,
try
语句块中异常发生点后的剩余语句不会被执行;Python 解释器搜索对应的异常处理器,一旦找到就开始执行处理器中的代码;
如果没有找到合适的异常处理器, 那么异常就自底向上移交给调用者去处理(最顶层也无法处理则退出程序)。
10.3.2 带有多个 except
的try
语句
def safe_float(obj): try: retval = float(obj)
except ValueError: retval = 'could not convert non-number to float' except TypeError: retval = 'object type cannot be converted to float' return retval
10.3.3 同时处理多个异常的 except
语句
def safe_float(obj): try: retval = float(obj)
except (ValueError, TypeError): retval = 'argument must be a number or numeric string' return retval
10.3.4 异常的层次结构
|- BaseException |- KeyboardInterrupt |- SystemExit |- Exception |- (all other current built-in exceptions)
10.3.5 异常参数
# multiple exceptions
except (Exception1,..., ExceptionN)[, reason]: suite_for_Exception1_to_ExceptionN_with_Argument
上面的reason
是一个包含异常代码诊断信息的类实例,异常参数自身会组成一个元组并存储为reason
的属性args
。
获取异常的错误信息:
def safe_float(object): try: retval = float(object)
except (ValueError, TypeError), e: retval = str(e)
return retval
10.3.6 else
子句
else
子句执行的条件:try
范围中的所有代码没有引发异常
try: module.function
except:
print 'error'
else:
print 'success'
10.3.7 finally
子句
finally
子句:是无论是否有异常发生,都会执行的一段代码。
10.3.8 try-finally
语句
try: A
finally: B
当在 try
范围中产生一个异常时会立即跳转到finally
语句,当finally
语句执行完毕后会继续向上一层引发异常。
10.3.9 try-except-else-finally
语句
try: A
except MyException: B
else: C
finally: D
结果:A-C-D[正常] 或 A-B-D[异常]
10.3.10 try
语句后跟子句的总结
有
else
子句则一定要有except
子句finally
子句前可以没有except
或else
子句
10.4 上下文管理
10.4.1 with 语句
with
语句用来简化try-except-finally
语句,仅工作于支持上下文管理协议(context management protocol)的对象。
with open('/etc/passwd', 'r') as f:
for eachLine in f:
# ...do stuff with eachLine or f... #
10.4.2 上下文管理协议
上下文管理器必须实现的方法:
__context__
:提供上下文对象__enter__
:完成 with 语句块执行前的准备工作__exit__
:当 with 语句块执行结束调用
contextlib
模块的functions/decorators
,可以方便地创建上下文管理器。
10.5 字符串作为异常(过时)
Python 1.5 前,标准的异常是基于字符串实现的,但是从现在起建议使用异常类。
10.6 触发异常
通过 raise
语句,可手动触发异常。
语法:raise [SomeException [, args [, traceback]]]
SomeException
:字符串/异常类/实例args
:异常的参数是一个元组traceback
:用于exception—normally的追踪对象
例子:
try:
raise IOError, 'raise test'
except IOError, e:
print e
10.7 断言
assert
可用于触发异常:如果断言成功不采取任何措施,否则触发AssertionError
。
语法:assert expr[, args]
实例:
try: assert 1 == 0, 'One does not equal zero silly!'
except AssertionError, args:
print '%s: %s' % (args.__class__.__name__, args)
# AssertionError: One does not equal zero silly!
断言的原理:
def assert(expr, args=None): if _ _debug_ _ and not expr:
raise AssertionError, args
10.8 标准异常
所有的标准/内建异常、自定义异常都是继承自根异常 BaseException
。
(其他信息见 10.2 Python中的常见异常 和 10.3.4 异常的层次结构)
10.9 使用异常的目的与好处
目的:为了使程序足够健壮,可以处理理应用级别的错误(不至于灾难性地影响其执行环境),并提供用户级别的错误信息。
好处:异常不仅简化代码,而且简化整个错误管理体系。
10.10 异常和 sys 模块
通过 sys
模块中exc_info
获取异常信息。
import sys
try: float('abc123')
except: exc_tuple = sys.exc_info
print exc_tuple
""" (<type 'exceptions.ValueError'>, ValueError('could not convert string to float: abc123',), <traceback object at 0x0000000002AB3848>) """
sys.exc_info
得到的元组是:
exc_type
:异常类exc_value
:异常类的实例exc_traceback
:追踪对象,提供了发生异常的上下文,包含代码的执行帧、异常发生时的行号等信息。
10.11 异常相关的标准库
exceptions
:内建异常
contextlib
:使用with
语句的上下文对象工具
sys
:包含各种异常相关的对象和函数(见sys.ex*
)
本章与其他语言相比
try-except
后面可以跟else
子句,而在Java里没有这样的语法。Python的
with
语法简化了try-except-finally
,在Java8里类似try-catch
本章课后习题
10–1. 引发异常. 以下的哪个因素会在程序执行时引发异常? 注意这里我们问的并不是异常的原因.
a) 用户
b) 解释器
c) 程序
d) 以上所有
e) 只有 b) 和 c)
f) 只有 a) 和 c)
10–2. 引发异常. 参考上边问题的列表, 哪些因素会在运行交互解释器时引发异常?
I/O(文件不存在、读取错误)、系统错误、边界值溢出等。
10–3. 关键字. 用来引发异常的关键字有那些?
raise
、try
10–4. 关键字. try-except 和 try-finally 有什么不同?
try-except
:当try
代码块运行出错,except
代码块才会被执行(匹配到相应异常时),处理完异常后try-except
后的代码会继续往下执行;try-finally
:不管try
代码块是否出现异常,最终都会执行finally
里的代码。如果出现异常,执行完finally
代码块后,异常会继续往上层抛。
10–5. 异常. 下面这些交互解释器下的 Python 代码段分别会引发什么异常(参阅表 10.2 给出的内建异常清单):
(a) >>> if 3 < 4 then: print '3 IS less than 4!' (b) >>> aList = ['Hello', 'World!', 'Anyone', 'Home?'] >>> print 'the last string in aList is:', aList[len(aList)] (c) >>> x (d) >>> x = 4 % 0 (e) >>> import math >>> i = math.sqrt(-1)
(a) SyntaxError
(b) IndexError
(c) NameError
(d) ZeroDivisionError
(e) ValueError
10–6. 改进的 open. 为内建的 open 函数创建一个封装. 使得成功打开文件后, 返回文件句柄; 若打开失败则返回给调用者 None , 而不是生成一个异常. 这样你打开文件时就不需要额外的异常处理语句.
def myopen(path, access='r'): try: f = open(path, access)
except (IOError, ValueError):
return None return f
if _ _name_ _ == '__main__':
print myopen("test.txt", 'abc')
10–7. 异常. 下面两段 Python 伪代码 a) 和 b) 有什么区别? 考虑语句 A 和 B 的上下文环境.
(a) try: statement_A
except . . .: . . .
else: statement_B (b) try: statement_A statement_B
except . . .: . . .
情况1:如果statement_A
执行出错,则(a)和(b)都不会执行statement_B
;
情况2:如果statement_A
执行正常,则(a)和(b)都会执行statement_B
——但是,如果statement_B
的执行也出错,(b)可以在except
子句里捕获到异常并做处理,而(a)则只能将statement_B
产生的异常抛给上层调用者。
10–8. 改进的 raw_input . 本章的开头, 我们给出了一个”安全”的 float 函数, 它建立在内建函数 float 上, 可以检测并处理 float 可能会引发的两种不同异常. 同样, raw_input 函数也可能会生成两种异常, EOFError (文件末尾 EOF, 在 Unix 下是由于按下了 Ctrl+D 在 Dos 下是因为 Ctrl+Z) 或 是 KeyboardInterrupt (取消输入 , 一 般是由于按下了Ctrl+C). 请创建一个封装函数 safe_input , 在发生异常时返回 None .
def safe_input(msg): data = None try: data = raw_input(msg)
except (EOFError, KeyboardInterrupt):
pass
return data
if _ _name_ _ == '__main__':
print safe_input("please input: ")
10–9. 改进的 math.sqrt. math 模块包含大量用于处理数值相关运算的函数和常量. 不幸的是, 它不能识别复数, 所以我们创建了 cmath 模块来支持复数相关运算. 请创建一个 safe_sqrt 函数, 它封装 math.sqrt 并能处理负值, 返回一个对应的复数.
import math, cmath
def safe_sqrt(num): try: result = math.sqrt(num)
except ValueError: result = cmath.sqrt(num)
return result
if _ _name_ _ == '__main__':
print safe_sqrt(123)
print safe_sqrt(-123)
复盘总结
异常的处理使得程序具有在运行时管理错误的能力,即使发生问题也可以采取可靠的补救措施。(领域间找关联:类比S君描述行动系统里的“抗风险地做”)
10.3.10那一小节里,作者有个地方写错:“这一节最重要的是无论你选择什么语法,你至少要有一个 except 子句,而 else 和 finally 都是可选的.”。提醒我们在学习的时候,要带着批判和怀疑的眼光去学习,而不是一味地接受别人的观点。
ScalersTalkID:scalerstalk
本微信公众号作者Scalers,游走在口译世界的IT从业者。微信公众号ScalersTalk,网站ScalersTalk.com,口译100小时训练计划群C 456036104
成长会是由Scalers发起的面向成长、实践行动,且凝聚了来自全球各地各行各业从业者的社群。有意入会者请和Scalers直接联系,我和其他会员会和你直接交流关于成长行动等各方面的经验教训。2016年成长会持续招募中,参见做能说会写的持续行动者:ScalersTalk成长会2016年会员计划介绍(2016.3更新)
猜你喜欢
- 2025-01-21 使用Python操作Jenkins(创建,构建,获取Job日志和报告)
- 2025-01-21 「python小脚本」监听日志文件异常数据发送告警短信
- 2025-01-21 告别千篇一律,Python打印彩色日志的方法!
- 2025-01-21 用Python写一个MacOS的系统通知
- 2025-01-21 Python 自制日志装饰器
- 2025-01-21 Python如何在日志中隐藏明文密码
- 2025-01-21 Kubernetes日志采集ELK|收集Python项目日志并展示
- 2025-01-21 Python接口自动化核心模块 - 数据库操作和日志
- 2025-01-21 python模块之 loguru 日志模块
- 2025-01-21 小python帮大忙——使用python生成word格式的工作日志
- 05-25Python 3.14 t-string 要来了,它与 f-string 有何不同?
- 05-25Python基础元素语法总结
- 05-25Python中的变量是什么东西?
- 05-25新手常见的python报错及解决方案
- 05-2511-Python变量
- 05-2510个每个人都是需要知道Python问题
- 05-25Python编程:轻松掌握函数定义、类型及其参数传递方式
- 05-25Python基础语法
- 257℃Python短文,Python中的嵌套条件语句(六)
- 257℃python笔记:for循环嵌套。end=""的作用,图形打印
- 256℃PythonNet:实现Python与.Net代码相互调用!
- 251℃Python操作Sqlserver数据库(多库同时异步执行:增删改查)
- 251℃Python实现字符串小写转大写并写入文件
- 106℃原来2025是完美的平方年,一起探索六种平方的算吧
- 90℃Python 和 JavaScript 终于联姻了!PythonMonkey 要火?
- 81℃Ollama v0.4.5-v0.4.7 更新集合:Ollama Python 库改进、新模型支持
- 最近发表
- 标签列表
-
- python中类 (31)
- python 迭代 (34)
- python 小写 (35)
- python怎么输出 (33)
- python 日志 (35)
- python语音 (31)
- python 工程师 (34)
- python3 安装 (31)
- python音乐 (31)
- 安卓 python (32)
- python 小游戏 (32)
- python 安卓 (31)
- python聚类 (34)
- python向量 (31)
- python大全 (31)
- python次方 (33)
- python桌面 (32)
- python总结 (34)
- python浏览器 (32)
- python 请求 (32)
- python 前端 (32)
- python验证码 (33)
- python 题目 (32)
- python 文件写 (33)
- python中的用法 (32)