1. 异常
1.1 程序在运行过程中,不可避免的会出现一些错误,比如:使用了没有赋值的变量,除0…,这些错误在程序中,称为异常
1.2 程序在运行中,出现异常将会导致程序立即终止,异常后面的代码全部不会执行
2. 异常处理
2.1 程序运行时出现异常,目的并不是让程序直接终止,而是希望编写代码对异常进行处理
2.2 处理方法:try语句,将可能出错的代码放到try语句,这样代码如果没有错误,就会正常执行;如果出现错误,会执行except子句中的代码块,可以通过代码处理异常,避免因为异常导致整个程序的终止
try:
代码块(可能出现错误的语句)
except 异常类型 as 异常名:
代码块(出现错误后的处理方式)
except 异常类型 as 异常名:
代码块(出现错误后的处理方式)
else:
代码块(没出错时要执行的语句)
finally:
代码块(无论有没有异常,都会执行)
1. try必须有,else有没有都可以,except和finally至少有一个
2. 如果except后面不跟任何内容,会捕获所有异常;如果except后面跟一个异常的类型,那么只会捕获该类型的异常;例如NameError,其中Exception是所有异常的父类,所以如果except后面跟exception,也会捕获所有的异常
3. 可以在异常类后面跟一个as xx,此时xx就是异常对象
4. finally 中的代码块一定会执行,应用场景:写文件时遇到报错,将保存文件的代码放到 finally 中,这样不管有没有遇到报错,都会执行保存文件操作
3. 例如
3.1 简单结构
print("hello")
try:
print(10/0)
except:
print("哦哦,出现错误了")
else:
print("程序正常执行")
print("nice to meet you")
>>>hello
>>>哦哦,出现错误了
>>>nice to meet you
3.2 复杂结构
l = []
print("异常出现前")
try:
#l[10]
print(c)
print(10/0)
except NameError:
print("出现 NameError 问题")
except IndexError:
print('出现 IndexError 问题')
except TypeError as t:
print("出现TypeError 问题",t,type(t))
except Exception as e:
print("未知错误",e,type(e))
finally:
print("无论有没有异常,都会执行")
print("异常出现后")
>>>异常出现前
>>>出现 NameError 问题
>>>无论有没有异常,都会执行
>>>异常出现后
4. 异常的传播
4.1 当函数中出现异常时,如果在函数中对异常进行了处理,则异常不会再继续传播;如果函数中没有对异常进行处理,则异常会向函数调用处传播;如果函数调用处处理了异常,则不再传播;如果没有处理则继续向调用处传播,直到传递到全局作用域(主模块);如果依然没有处理,则程序终止,并且显示异常信息
4.2 例如
def fn():
print("this is fn")
print(a)
def fn2():
print("this is fn2")
fn()
def fn3():
print("this is fn3")
fn2()
fn3()
>>>this is fn3
>>>this is fn2
>>>this is fn
>>>Traceback (most recent call last):
>>> File "D:/1/MyPython/selfStudy/practice/Python_10_TryExcept.py", line 23, in <module>
>>> fn3()
>>> File "D:/1/MyPython/selfStudy/practice/Python_10_TryExcept.py", line 22, in fn3
>>> fn2()
>>> File "D:/1/MyPython/selfStudy/practice/Python_10_TryExcept.py", line 19, in fn2
>>> fn()
>>> File "D:/1/MyPython/selfStudy/practice/Python_10_TryExcept.py", line 15, in fn
>>> print(a)
>>>NameError: name 'a' is not defined
4.3 异常信息种类:当程序运行中出现异常后,所有的异常信息会被保存到一个专门的异常对象中,而异常传播时,实际上就是异常对象抛给了调用处。比如:ZeroDivisionError类的对象用来处理除0的异常;NameError类的对象用来处理变量错误的异常。。。
5. 自定义异常
5.1 例如:
def add(a,b):
# 如果a或b中有负数,就向调用处抛出异常
if a < 0 or b < 0:
# raise用于向外部抛出异常,后面可以跟异常类,或异常类的实例
# 抛出异常的目的,是告诉调用者这里调用时出现问题,希望你自己处理
raise Exception("a,b两个参数必须大于0")
result = a + b
return result
print(add(-12,3))
>>>Traceback (most recent call last):
>>> File "D:/1/MyPython/selfStudy/practice/Python_10_TryExcept.py", line 51, in <module>
>>> print(add(-12,3))
>>> File "D:/1/MyPython/selfStudy/practice/Python_10_TryExcept.py", line 47, in add
>>> raise Exception("a,b两个参数必须大于0")
>>>Exception: a,b两个参数必须大于0
5.2 自定义异常类
# 以下异常类实现了传入的实参必须大于0,否则报错
class MyError(Exception):
pass
def add(a,b):
if a < 0 or b < 0:
raise MyError("自定义异常类")
result = a + b
return result
print(add(-12,3))
>>>Traceback (most recent call last):
>>> File "D:/1/MyPython/selfStudy/practice/Python_10_TryExcept.py", line 54, in <module>
>>> print(add(-12,3))
>>> File "D:/1/MyPython/selfStudy/practice/Python_10_TryExcept.py", line 50, in add
>>> raise MyError("自定义异常类")
>>>__main__.MyError: 自定义异常类