-
什么是异常?
异常就是与正常情况不同,程序在执行过程中出现错误,导致无法执行完毕。异常其实就是代码执行过程中出错。
-
常见的一些异常
-
AttributeError 试图访问一个对象没有的属性,比如foo.x,但是foo没有属性x
-
IOError 输入/输出异常;无法打开文件或无法读写
-
ImportError 无法引入模块或包;基本上是路径问题或名称错误
-
SyntaxError Python语法错误异常,代码不能编译
-
IndentationError 缩进异常;代码没有正确缩进
-
IndexError 下标索引超出序列边界,比如当x只有三个元素,却试图访问x[5]
-
KeyError 试图访问字典里不存在的键
-
KeyboardInterrupt Ctrl+C被按下
-
NameError 使用一个还未被赋予对象的变量
-
TypeError 传入对象类型与要求的不符合
-
UnboundLocalError 试图访问一个还未被设置的局部变量,基本上是由于另有一个同名的全局变量,
导致你以为正在访问它 -
ValueError 传入一个调用者不期望的值,即使值的类型是正确的
-
-
异常组成的三个部分
追踪信息、异常类型、异常的值
异常可以由发生的时间不同分为两类:
-
语法检测异常:解释器解释python语法时出现异常。必须在程序运行前改正。
-
运行时异常:已经通过了语法检测,在执行期间发生异常。(逻辑错误)
几种不同的异常处理的语法
1.如果错误发生的条件是可以预知的,我们需要用if进行处理:在错误发生之前进行预防
AGE=10 while True: age=input('>>: ').strip() if age.isdigit(): #只有在age为字符串形式的整数时,下列代码才不会出错,该条件是可预知的 age=int(age) if age == AGE: print('you got it') break else: print('you are wrong') break else: print('请输入数字~~')
2.如果发生的错误条件是不可以预知的,我们就需要用到try......except的几种用法:
在错误发生之后进行处理。
-
try.....except
try:
被检测的代码块
except 异常类型:
try中一旦检测到异常,就执行这个位置的逻辑#这种方式只能用来处理指定的异常情况,如果非指定异常则无法处理。如果是非指定的异常类型会直接报错。 try: print('starting') a = int('deng') # 这个位置很明显是有逻辑错误的 print('ending') # 未执行这行代码 except ValueError: # 检测到异常后执行了下面代码 print('发生了ValueError') ## starting 发生了ValueError
-
多分枝
try:
被检测的代码块
except 异常类型1:
try中一旦检测到异常,就执行这个位置的逻辑except 异常类型2:
try中一旦检测到异常,就执行这个位置的逻辑try: print('starting') l = [] print(l[1]) #这个位置发生了IndexError int('deng') #未执行 print('ending') except ValueError: #未执行 print('发生了ValueError') except IndexError: #检测到异常,执行下面代码 print('发生了IndexError') ### starting 发生了IndexError
-
同一分支检测多种异常
try:
被检测的代码块
except(多种异常类型):
try一旦检测到异常,就执行这个位置的逻辑
try: print('starting') l = [] print(l[1]) #可以看出代码到这个位置就停止了 print('ending') int('deng') except (ValueError,IndexError): #只要检测到异常就运行下面代码 print('发生了ValueError') ### starting 发生了ValueError
-
万能异常1
try:
被检测的代码块
except Exception:
try一旦检测到异常,就执行这个位置的逻辑
#这种方式尽量不要用,会导致程序既不报错,也不会正常运行,无法定位报错位置。 try: print('starting') l = [] print(l[1]) print('ending') int('deng') except Exception: print('going')
-
万能异常2
try:
被检测的代码块
except Exception as e:
try一旦检测到异常,就执行这个位置的逻辑
try: print('starting') l = [] print(l[1]) print('ending') int('deng') except Exception as e: #通过给异常取别名,来获取异常对象,对象中包含错误信息 print('going') print(e) print(type(e)) ### starting going list index out of range <class 'IndexError'>
我们可以看到e是一个类,所有的异常类都是Exception的子类。
万能异常与多分枝异常处理区别。
1.如果你想要的效果是,无论出现什么异常,我们统一丢弃,或者使用同一段代码逻辑去处理他们,那么骚年,大胆的去做吧,只有一个Exception就足够了。
2.如果你想要的效果是,对于不同的异常我们需要定制不同的处理逻辑,那就需要用到多分支了。 -
明确类型与万能异常2合用
try:
被检测的代码块
except 异常类型1:
try中一旦检测到异常,就执行这个位置的逻辑except 异常类型2:
try中一旦检测到异常,就执行这个位置的逻辑except Exception as e:
try一旦检测到异常,就执行这个位置的逻辑
#明确类型应该放在万能类型前才能正常使用 try: print('starting') l = [] print(l[1]) print('ending') int('deng') except ValueError: print('值发生错误。。。') except ZeroDivisionError: print('被除数不能为0') except Exception as e: print('going')
-
try:
被检测的代码块
except 异常类型1:
try中一旦检测到异常,就执行这个位置的逻辑except 异常类型2:
try中一旦检测到异常,就执行这个位置的逻辑except Exception as e:
try一旦检测到异常,就执行这个位置的逻辑
else:
代码体中没有异常执行这个位置的逻辑
try: print('starting') l = [] print(l[1]) print('ending') int('deng') except ValueError: print('值发生错误。。。') except ZeroDivisionError: print('被除数不能为0') except Exception as e: print('going') else: print('这段代码没有问题')
-
try:
被检测的代码块
except 异常类型1:
try中一旦检测到异常,就执行这个位置的逻辑except 异常类型2:
try中一旦检测到异常,就执行这个位置的逻辑except Exception as e:
try一旦检测到异常,就执行这个位置的逻辑
else:
代码体中没有异常执行这个位置的逻辑
finally:
代码体不管是否有异常,最终都执行该部分逻辑
try: print('starting') l = [] print(l[1]) print('ending') int('deng') except ValueError: print('值发生错误。。。') except ZeroDivisionError: print('被除数不能为0') except Exception as e: print('going') else: print('这段代码没有问题') finally: print('这是备用方案~~') print('这是关闭文件操作')
-
主动触发抛出异常
try:
raise TypeError('类型错误')
except Exception as e:
print(e)#自定义异常 class LoginException(Exception): pass def login(): name = input('name>>:').strip() pwd = input('password>>:').strip() if name == 'deng' and pwd == '123': print('login') else: raise LoginException('用户名密码错误!~') login()
-
断言 assert
在程序中,有一段代码体,要执行必须保证某个条件必须成立,类似于if判断,但是断言不用将代码体缩进至其内部。
#使用if判断 l = ['ming','deng'] if l: print('nihaoa ') #使用断言 assert l print('nihaom') print(l[0])
总结:异常处理的作用
1:把错误处理和真正的工作分开来
2:代码更易组织,更清晰,复杂的工作任务更容易实现;
3:毫无疑问,更安全了,不至于由于一些小的疏忽而使程序意外崩溃了。