飞翔飞翔
主页
  • 计算机基础

    • TCP/IP协议
    • Linux命令
  • 数据库

    • SQL教程
  • 编程语言

    • C语言
    • Python2
    • Python3
  • 数据格式

    • JSON教程
  • 工具

    • Markdown指南
  • Git

    • GitFlow
  • Quartz

    • Quartz教程
  • Java

    • Java设计模式
  • 缓存

    • Redis教程
联系
阿里云
主页
  • 计算机基础

    • TCP/IP协议
    • Linux命令
  • 数据库

    • SQL教程
  • 编程语言

    • C语言
    • Python2
    • Python3
  • 数据格式

    • JSON教程
  • 工具

    • Markdown指南
  • Git

    • GitFlow
  • Quartz

    • Quartz教程
  • Java

    • Java设计模式
  • 缓存

    • Redis教程
联系
阿里云
  • 学习路径
  • 第1章 认识Python

    • Python 历史与特点
    • Python 2 与 Python 3 的核心差异
    • 安装与运行 Python 2.7.18
    • 编码规范 PEP 8
  • 第2章 基础语法

    • 变量与对象
    • 数字类型
    • 字符串 str
    • Unicode 字符串
    • 运算符
    • 空值 None
  • 第3章 流程控制

    • if 条件语句
    • if-else 条件语句
    • if-elif-else 多分支
    • 条件表达式(三元运算符)
    • while 循环
    • for 循环
    • range 与 xrange
    • 循环控制:break、continue、pass
    • 循环 else 子句
  • 第4章 数据结构

    • 列表基础
    • 列表方法
    • 列表推导式
    • 元组
    • 字典基础
    • 字典方法
    • 字典循环技巧
    • 集合
    • 序列解包
    • 序列比较
  • 第5章 函数

    • 定义函数
    • 参数传递机制
    • 默认参数
    • 关键字参数
    • 可变参数
    • Lambda 表达式
    • 文档字符串
    • 函数对象
  • 第6章 模块与包

    • import 导入
    • 模块搜索路径
    • name 与主程序
    • 编译文件 .pyc 与 .pyo
    • 包结构
    • dir() 函数
  • 第7章 文件与IO

    • 打开与关闭文件
    • 文件读写方法
    • with 上下文管理器
    • 格式化输出:% 操作符
    • 格式化输出:str.format()
    • JSON 序列化
  • 第8章 面向对象

    • 类定义与实例化
    • init 构造方法
    • 类变量与实例变量
    • 方法调用与 self
    • 继承基础
    • 多重继承
    • 新式类与旧式类
    • 私有变量与名称改写
    • 属性装饰器 property
    • 类方法与静态方法
    • 魔术方法
    • 空类与数据记录
  • 第9章 异常处理

    • 异常类型
    • try-except
    • try-except-else-finally
    • 抛出异常 raise
    • 自定义异常
    • with 语句与上下文管理器
  • 第10章 迭代器与生成器

    • 迭代器协议
    • 生成器函数
    • 生成器表达式
    • itertools模块
  • 第11章 标准库精要

    • os模块
    • sys模块
    • datetime模块
    • re模块
    • json模块
    • collections模块
    • math与random模块
    • urllib2与网络请求
    • subprocess与命令执行
    • threading与并发
    • unittest与测试
    • 虚拟环境与包管理
  • 第12章 工程实践

    • 调试技巧
    • 性能分析
    • 文档与注释
    • 下一步学习

异常类型

异常(Exception)是 Python 处理错误的机制。当程序遇到无法继续执行的情况时,会"抛出"一个异常对象。如果异常没有被捕获,程序会终止并打印错误信息。Python 2 内置了丰富的异常层次结构,理解这些异常类型是写出健壮代码的基础。

异常层次结构

所有内置异常都继承自 BaseException,但用户自定义异常通常继承自 Exception:

BaseException
 ├── SystemExit              # sys.exit() 引发
 ├── KeyboardInterrupt       # Ctrl+C 引发
 ├── GeneratorExit           # 生成器关闭时引发
 └── Exception               # 所有内置非系统退出异常的基类
      ├── ArithmeticError
      │    ├── ZeroDivisionError    # 除零
      │    └── OverflowError        # 数值溢出
      ├── LookupError
      │    ├── IndexError           # 序列索引越界
      │    └── KeyError             # 字典键不存在
      ├── TypeError                 # 类型错误
      ├── ValueError                # 值错误
      ├── AttributeError            # 属性不存在
      ├── IOError                   # 输入输出错误
      ├── ImportError               # 导入失败
      ├── NameError                 # 名称未定义
      │    └── UnboundLocalError    # 局部变量未绑定
      ├── SyntaxError               # 语法错误
      └── RuntimeError              # 通用运行时错误

常见异常详解

SyntaxError:代码语法不正确,无法编译

if x > 0          # SyntaxError: invalid syntax(缺少冒号)
    print x

SyntaxError 在代码编译时抛出,无法被 try-except 捕获(因为异常处理本身也是代码,需要编译后才能执行)。

NameError:使用了未定义的变量名

print undefined_var     # NameError: name 'undefined_var' is not defined

TypeError:操作或函数应用于不适当类型的对象

print len(42)           # TypeError: object of type 'int' has no len()
print "hello" + 5       # TypeError: cannot concatenate 'str' and 'int' objects

ValueError:操作或函数接收到具有正确类型但不适当的值

print int("hello")      # ValueError: invalid literal for int() with base 10: 'hello'

IndexError:序列索引超出范围

nums = [1, 2, 3]
print nums[10]          # IndexError: list index out of range

KeyError:字典中不存在指定的键

d = {"a": 1}
print d["b"]            # KeyError: 'b'

AttributeError:对象没有该属性

s = "hello"
print s.append("!")     # AttributeError: 'str' object has no attribute 'append'

ZeroDivisionError:除零

print 1 / 0             # ZeroDivisionError: integer division or modulo by zero

IOError:输入输出操作失败

f = open("nonexistent.txt", "r")    # IOError: [Errno 2] No such file or directory

ImportError:导入模块失败

import nonexistent_module           # ImportError: No module named nonexistent_module

异常对象的信息

异常对象包含错误信息,可以通过 except 捕获后访问:

try:
    1 / 0
except ZeroDivisionError as e:
    print type(e)       # <type 'exceptions.ZeroDivisionError'>
    print e             # integer division or modulo by zero
    print e.args        # ('integer division or modulo by zero',)

e.args 是异常构造时传入的参数元组。大多数内置异常只有一个字符串参数。

异常与错误处理哲学

Python 倡导 EAFP(Easier to Ask for Forgiveness than Permission):

# EAFP:先尝试,出问题再处理
try:
    value = d[key]
except KeyError:
    value = default

# LBYL:先检查,再操作(Look Before You Leap)
if key in d:
    value = d[key]
else:
    value = default

EAFP 在 Python 中通常更快、更简洁,因为字典查找是 O(1),两次查找(in + [])比一次查找加异常处理更慢。但对于高频率的简单操作,LBYL 可能更合适。

异常的性能

异常处理本身开销很小,但抛出异常很昂贵:

import timeit

# 使用异常(不触发)
t1 = timeit.timeit('try:\n    x = 1\nexcept:\n    pass', number=1000000)

# 使用异常(触发)
t2 = timeit.timeit('try:\n    x = 1/0\nexcept:\n    pass', number=100000)

print t1, t2        # 不触发很快,触发慢约 100 倍

因此,不要用异常控制正常流程——异常应该用于"异常情况"。

下一页
try-except