目录
异常和错误
异常机制本质
异常指程序运行过程中出现的非正常现象,例如用户输入错误、除数为零、需要处理的文件不存在、数组下标越界等。
所谓异常处理,就是指程序在出现问题时依然可以正确的执行剩余的程序,而不会因为异常而终止程序执行。
Python 中内建异常类的继承层次
解决异常问题的态度
1.不慌张,细看信息,定位错误。看清楚报的错误信息,并定位发生错误的地方;
2.百度并查看十个相关帖子。将异常类信息进行百度,至少查看十个以上的相关帖子;
3.以上两步仍然无法解决,找老师和同学协助解决。
异常解决的关键:定位
当发生异常时,解释器会报相关的错误信息,并会在控制台打印出相关错误信息。我们只需按照从上到下的顺序即可追溯(Trackback)错误发生的过程,最终定位引起错误的那一行代码。
try…一个except 结构
try 块包含着可能引发异常的代码,except 块则用来捕捉和处理发生的异常。执行的时候,如果try 块中没有引发异常,则跳过except 块继续执行后续代码;执行的时候,如果try 块中发生了异常,则跳过try 块中的后续代码,跳到相应的except 块中处理异常;异常处理完后,继续执行后续代码。
# 测试简单的 0不能做除数异常
try:
print("step1")
a = 3/0
print("step2")
except BaseException as e:
print("step3")
print(e)
print("step4")
''' 执行结果: step1 step3 division by zero step4 '''
try…多个except 结构
从经典理论考虑,一般建议尽量捕获可能出现的多个异常(按照先子类后父类的顺序),并且针对性的写出异常处理代码。为了避免遗漏可能出现的异常,可以在最后增加BaseException。
try:
a = input("请输入被除数:")
b = input("请输入除数:")
c = float(a)/float(b)
print(c)
except ZeroDivisionError:
print("异常:除数不能为0")
except TypeError:
print("异常:除数和被除数都应该为数值类型")
except NameError:
print("异常:变量不存在")
except BaseException as e:
print(e)
print(type(e))
''' 执行结果: 请输入被除数:10 请输入除数:0 异常:除数不能为0 '''
try…except…else 结构
如果try 块中没有抛出异常,则执行else 块。如果try 块中抛出异常,则执行except 块,不执行else 块。
try:
a = input("请输入被除数:")
b = input("请输入除数:")
c = float(a)/float(b)
except BaseException as e:
print(e)
else:
print("除的结果是:",c)
''' 发生异常的执行情况(执行except 块,没有执行else): 请输入被除数:5 请输入除数:0 float division by zero 没有发生异常的执行情况(执行完try 块后,执行else): 请输入被除数:10 请输入除数:5 除的结果是:2.0 '''
try…except…finally 结构
try…except…finally 结构中,finally 块无论是否发生异常都会被执行,通常用来释放try 块中申请的资源。
try:
a = input("请输入一个被除数:")
b = input("请输入一个除数:")
c = float(a)/float(b)
except BaseException as e:
print(e)
else:
print(c)
finally:
print("我是finally 中的语句,无论发生异常与否,都执行!")
print("程序结束!")
''' 执行结果如下: 请输入被除数:10 请输入除数:0 float division by zero 我是finally 中的语句,无论是否发生异常都执行 '''
return 语句和异常处理问题
return 有两种作用:结束方法运行、返回值。我们一般不把return 放到异常处理结构中,而是放到方法最后。
def test01():
print("step1")
try:
x = 3/0
except:
print("step2")
print("异常:0 不能做除数")
finally:
print("step4")
print("step5")
return "e"
# 一般不要将return 语句放到try、except、else、finally 块中,会发生一些意想不到的错误。建议放到方法最后。
print(test01())
''' 执行结果: step1 step2 异常:0 不能做除数 step4 step5 e '''
常见异常的解决
Python 中的异常都派生自BaseException 类。
1.SyntaxError:语法错误
2.NameError:尝试访问一个没有申明的变量
3.ZeroDivisionError:除数为0 错误(零除错误)
4.ValueError:数值错误
5.TypeError:类型错误
6.AttributeError:访问对象的不存在的属性
7.IndexError:索引越界异常
8.KeyError:字典的关键字不存在
with 上下文管理
with 上下文管理可以自动管理资源,在with 代码块执行完毕后自动还原进入该代码之前的现场或上下文。不论何种原因跳出with 块,不论是否有异常,总能保证资源正常释放。
with open("d:/bb.txt") as f:
for line in f:
print(line)
''' 执行结果: gaoqi sxt baizhan '''
trackback 模块
# 使用 traceback 将异常信息写入日志文件
# coding=utf-8
import traceback
try:
print("step1")
num = 1/0
except:
with open("d:/a.log","a") as f:
traceback.print_exc(file=f)
自定义异常类
程序开发中,有时候我们也需要自己定义异常类。自定义异常类一般都是运行时异常,通常继承Exception 或其子类即可。命名一般以Error、Exception 为后缀。自定义异常由raise 语句主动抛出。
一个python的文件有两种使用的方法,第一是直接作为脚本执行,第二是import到其他的python脚本中被调用(模块重用)执行。
因此if __name__ == ‘__main__’: 的作用就是控制这两种情况执行代码的过程,在if __name__ == ‘__main__’: 下的代码只有在第一种情况下(即文件作为脚本直接执行)才会被执行,而import到其他脚本中是不会被执行的。
#coding=utf-8
#测试自定义异常类
class AgeError(Exception): #继承Exception
def __init__(self,errorInfo):
Exception.__init__(self)
self.errorInfo = errorInfo
def __str__(self):
return str(self.errorInfo)+",年龄错误!应该在1-150 之间"
############测试代码################
if __name__ == "__main__": #如果为True,则模块是作为独立文件运行,可以执行测试代码
age = int(input("输入一个年龄:"))
if age<1 or age>150:
raise AgeError(age)
else:
print("正常的年龄:",age)
''' 执行结果如下: 输入一个年龄:200 Traceback (most recent call last): File "C:/Users/Administrator/PycharmProjects/mypro_exception/my10.py", line 16, in <module> raise AgeError(age) __main__.AgeError: 200,年龄错误!应该在 1-150 之间 '''
Pycharm 开发环境的调试
进行调试的核心是设置断点。程序执行到断点时,暂时挂起,停止执行。就像看视频按下停止一样,我们可以详细的观看停止处的每一个细节。
断点
进入调试视图
调试操作区
文章评论