本文共 2713 字,大约阅读时间需要 9 分钟。
一、概念
异常就是程序运行时发生错误的信号(在程序出现错误时,则会产生一个异常,若程序没有处理它,则会抛出该异常,程序的运行也随之终止
1、错误包括语法错误和逻辑错误
(1)语法错误
这种错误python解释器检测错误时不能通过,必须在程序执行前就改正错误
class Foo #语法错误 SyntaxError: invalid syntax
pass
(2)逻辑错误
1 2 3 4 | l = [ 'egon' , 'aa' ] l[ 3 ] #报错IndexError dic = { 'name' : 'egon' } dic[ 'age' ] #报错KeyError |
二、异常的种类
AttributeError #试图访问一个对象没有的树形,比如foo.x,但是foo没有属性x
IOError #输入/输出异常;基本上是无法打开文件
ImportError #无法引入模块或包;基本上是路径问题或名称错误
IndentationError #语法错误(的子类) ;代码没有正确对齐
IndexError #下标索引超出序列边界,比如访问不存在元素的下标
KeyError #试图访问字典里不存在的键
KeyboardInterrupt #Ctrl+C被按下
NameError #使用一个还未被赋予对象的变量
SyntaxError #Python代码非法,代码不能编译(个人认为这是语法错误,写错了)
TypeError #传入对象类型与要求的不符合
UnboundLocalError #试图访问一个还未被设置的局部变量,基本上是由于另有一个同名的全局变量,导致你以为正在访问它
ValueError #传入一个调用者不期望的值,即使值的类型是正确的
三、异常处理
为了保证程序的健壮性与容错性,即在遇到错误时程序不会崩溃,我们需要对异常进行处理
1、如果错误发生的条件是可预知的,需要用if进行处理
1 2 3 4 5 6 7 8 | AGE = 22 while True : age = input ( '>>: ' ).strip() if age.isdigit(): #只有在age为字符串形式的整数时,下列代码才不会出错,如果不是报错IndentationError: unindent does not match any outer indentation level age = int (age) if age = = AGE: print ( 'you got it' ) break |
2、如果错误发生的条件是不可预知的,则需要用到try...except进行处理
(1)多分支异常处理,判断属于哪一种异常类型
1 2 3 4 5 6 7 8 9 | s1 = 'hello' try : int (s1) except IndexError as e: print (e) except KeyError as e: print (e) except ValueError as e: print (e) |
(2)万能异常Exception,可以匹配到所有的异常
1 2 3 4 5 6 7 | s1 = 'hello' try : int (s1) except Exception as e: print (e) 如果你想要的效果是,无论出现什么异常,我们统一丢弃,或者使用同一段代码逻辑去处理他们,那么只有一个Exception就足够了。 如果你想要的效果是,对于不同的异常我们需要定制不同的处理逻辑,那就需要用到多分支了。 |
(3)异常的其他机构
1 2 3 4 5 6 7 8 9 10 11 12 13 | s1 = 'hello' try : int (s1) except IndexError as e: print (e) except KeyError as e: print (e) except ValueError as e: print (e) else : print ( 'try内代码块没有异常则执行我' ) finally : print ( '无论异常与否,都会执行该模块,通常是进行清理工作' ) |
(4)主动触发异常
1 2 3 4 5 6 7 8 9 10 11 12 | try : raise TypeError( '类型错误' ) except Exception as e: print (e) 当程序下部分用到上部分的结果时,可以判断上面的结果是否满足下部分所需要的条件,比如上面得到ip地址,下面验证可用性,可以在中间判断ip地址列表是否为空 ip_list = [ # '1.1.1.1:8080', # '1.1.1.2:8081', # '1.1.1.3:8082', ] if len (ip_list) = = 0 : raise TypeError |
(5)断言:assert 条件
1 2 3 4 5 6 7 | 上面的使用 if 条件实现主动触发异常,这里使用断言就可以实现相同的结果 ip_list = [ # '1.1.1.1:8080', # '1.1.1.2:8081', # '1.1.1.3:8082', ] assert len (ip_list) > 0 #当列表不为空时,正常执行下面的代码,否则报异常 |
(6)自定义异常
1 2 3 4 5 6 7 8 9 | class EgonException(BaseException): def __init__( self ,msg): self .msg = msg def __str__( self ): return self .msg try : raise EgonException( '类型错误' ) except EgonException as e: print (e) |
(7)总结try..except
1:把错误处理和真正的工作分开来
2:代码更易组织,更清晰,复杂的工作任务更容易实现;
3:毫无疑问,程序更安全了,不至于由于一些小的疏忽而使程序意外崩溃了
四、异常处理的应用场景
首先try...except是你附加给程序的一种异常处理的逻辑,与主要的工作是没有关系的,其次这种东西加的多了,会导致程序代码可读性变差
所以只有在错误发生的条件无法预知的情况下,才应该加上try...except处理异常
本文转自 宋鹏超 51CTO博客,原文链接:http://blog.51cto.com/qidian510/2066646,如需转载请自行联系原作者