一起养成写作习惯!这是我参与「掘金日新计划 · 4 月更文挑战」的第13天,点击查看活动详情。
一、初识异常
异常与异常处理:
- 异常既错误
- 异常会导致程序崩溃并停止运行
- 异常处理可以捕获到异常,将异常部位的程序进行处理使得程序继续正常运行
异常处理的结构由 try-except
代码块组成
try:
代码块 #被try关键字检查并保护的代码
except <异常类型> # try关键字检查的代码可能出现的异常的种类
代码块 # try关键字检查的代码出现异常后执行的代码
代码语言:javascript复制def upper(str_data):
new_str = str_data.upper()
return new_str
res = upper('hero')
print(res)
调用 upper 函数时,传入字符串,函数可以正常执行,如果传入数字则会报错
代码语言:javascript复制res_01 = upper(111)
print(res_01)
res_02 = upper('hello')
print(res_02)
错误代码后面的代码无论正确与否一旦出现异常,程序就会终止,这个时候就需要对这段可能会出错的代码进行异常处理,确保程序能正常运行。
代码语言:javascript复制def upper(str_data):
new_str = ''
try:
new_str = str_data.upper()
except:
print('程序执行出现问题')
return new_str
对函数中可能出现异常的代码进行了异常处理,程序可以正常执行。
捕获异常
可以对try-except代码中的异常进行捕获,再不确定是那种异常的情况下可以使用异常的基类Exception进行捕获
代码语言:javascript复制def divid(x, y):
res = 0
try:
res = x / y
except Exception as e:
print(e)
return res
res = divid(1, 2)
print(res)
res_01 = divid(1, 0)
print(res_01)
res_02 = divid(10, 2)
print(res_02)
不使用try-except异常处理,执行函数,查看控制台打印出的具体异常
代码语言:javascript复制def divid(x, y):
res = x / y
return res
可以看出控制塔打印的异常为 “ZeroDivisionError: division by zero”,如果可以确定具体的异常类型,也可以在except关键字后捕获具体的异常
代码语言:javascript复制def divid(x, y):
res = 0
try:
res = x / y
except ZeroDivisionError as e:
print(e)
return res
在对列表和字符串进行索引操作时还会出现索引超出范围异常,使用这个异常看能否不会divid函数中的异常
代码语言:javascript复制def divid(x, y):
res = 0
try:
res = x / y
except IndexError as e:
print(e)
return res
程序运行还是出错,这是因为异常类型不匹配,所以没有捕获到,在进行异常处理时如果可以确定具体的异常类型就使用具体的异常类型,如果不确定就是用基类异常,使用的异常不匹配程序仍然不能正确执行
捕获多个异常
可以通过连续添加except代码块的方式捕获多个异常。
代码语言:javascript复制try:
# 代码块
except Eexception1 as e1:
# 代码块
except EXception2 as e2:
# 代码块
except EXception3 as e3:
# 代码块
当 except 代码块有多个的时候,当异常类型匹配后,就不会继续往下捕获,也可以使用元组将所有待捕获的异常存起来。
代码语言:javascript复制try:
# 代码块
except (Eexception1, Exception2, Exception3) as e:
# 代码块
代码语言:javascript复制def divid(x, y):
res = 0
try:
res = x / y
except (IndexError, NameError, ZeroDivisionError) as e:
print(e)
return res
或者
代码语言:javascript复制def divid(x, y):
res = 0
try:
res = x / y
except IndexError as e:
print(e)
except NameError as e2:
print(e2)
except ZeroDivisionError as e3:
print(e3)
return res
二、常见异常类型
异常类型名称 | 异常类型描述 |
---|---|
Exception | 通用异常类型(基类) |
ZeroDivisionError | 整除0时出现的异常 |
AttributeError | 对象不含指定属性时出现的异常 |
IOError | IO流输入输出时出现的异常 |
KeyError | 没有指定的键时出现的异常 |
NameError | 指定变量不存在时出现的异常 |
SyntaxError | Python语法错误出现的异常 |
SystemError | 系统错误出现的异常 |
ValueError | 参数值异常 |
AttributeError
代码语言:javascript复制class Student:
name = None
age = 7
def __init__(self, name, age):
self.name = name
self.age = age
def __str__(self):
return 'Student[name={}, age={}]'.format(self.name, self.age)
stu = Student("stark", 44)
print(stu)
print(stu.address)
该异常类型为 AttributeError,既stu对象没有address属性,通过try-except对异常进行处理
代码语言:javascript复制try:
stu = Student("stark", 44)
print(stu)
print(stu.address)
except AttributeError as e:
print(e)
KeyError
代码语言:javascript复制dict_01 = {
'name': 'stark',
'age': 44
}
print(dict_01['address'])
对错误代码进行异常处理
代码语言:javascript复制try:
print(dict_01['address'])
except KeyError as e:
print(e)
IndexError
代码语言:javascript复制list_01 = [1, 3, 4, 9, 10]
list_01[10]
代码语言:javascript复制try:
list_01[10]
except IndexError as e:
print(e)
ValueError
ValueError 常出现在数据类型转换时
代码语言:javascript复制name = 'stark'
print(int(name))
代码语言:javascript复制try:
print(int(name))
except ValueError as e:
print(e)
TypeError
代码语言:javascript复制def add(x, y):
return x y
add()
代码语言:javascript复制try:
add()
except TypeError as e:
print(e)
三、finally 关键字
finally 关键字:
- 无论是否发生异常,一定会执行代码块
- 在函数中,即使在try或者except中执行了return后也依然会执行finally代码块中的内容
- try语法至少要伴随except或者finally中的一个
def divid(x, y):
try:
x / y
except Exception as e:
print(e)
finally:
return 'divid()函数执行完毕'
res = divid(1, 0)
print(res)
异常成功被捕获,并且执行了 finally 代码块中的内容,当 except 代码块中包含 return 时,finally 代码块依然会执行
代码语言:javascript复制def divid(x, y):
try:
x / y
except Exception as e:
return e
finally:
return 'divid()函数执行完毕'
res = divid(1, 0)
print(res)
四、自定义异常
raise关键字
raise 关键字可以抛出自定义的异常信息
代码语言:javascript复制raise ValueError('这是自定义的异常信息')
自定义异常
自定义异常类必须:
- 继承Exception基类
- 类构造函数中定义错误信息
def divid(x, y):
if isinstance(x, str) or isinstance(y, str):
raise Exception('参数类型不正确,不能为字符串')
return x y
res = divid(1, 2)
print(res)
res_01 = divid('stark', 'tony')
这里使用 raise 关键字输出错误信息,除此之外还可以自定义异常类来进行异常捕获
代码语言:javascript复制class NumberLimitError(Exception):
def __init__(self, message):
self.message = message
def divid(x, y):
if isinstance(x, str) or isinstance(y, str):
raise NumberLimitError('参数类型不正确,参数只能为数字类型')
return x y
try:
res = divid('stark', 'tony')
except NumberLimitError as e:
print(e)
自定义的异常能成功捕获异常信息。
五、断言
assert 可以对表达式进行判断,并输出指定的断言失败信息
代码语言:javascript复制assert expression, message
- expression:表达式一般是判断是否相等,或者判断某种数据类型的bool判断的语句
- message:断言失败时的错误提示信息
dict_01 = {
'name': 'stark',
'age': 40
}
assert len(dict_01) == 4, '断言失败, dict_01的长度不等于4'
使用 try-except
进行异常处理
try:
assert len(dict_01) == 4, '断言失败, dict_01的长度不等于4'
except AssertionError as e:
print(e)
如果 assert 成功,则不会实处任何信息
代码语言:javascript复制try:
assert len(dict_01) == 2, '断言失败, dict_01的长度不等于4'
except AssertionError as e:
print(e)