一、概念

异常就是程序运行时发生错误的信号(在程序出现错误时,则会产生一个异常,若程序没有处理它,则会抛出该异常,程序的运行也随之终止

1、错误包括语法错误和逻辑错误

(1)语法错误

这种错误python解释器检测错误时不能通过,必须在程序执行前就改正错误

class Foo             #语法错误 SyntaxError: invalid syntax

    pass

(2)逻辑错误

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进行处理

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)多分支异常处理,判断属于哪一种异常类型

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,可以匹配到所有的异常

s1 = 'hello'
try:
    int(s1)
except Exception as e:
    print(e)
如果你想要的效果是,无论出现什么异常,我们统一丢弃,或者使用同一段代码逻辑去处理他们,那么只有一个Exception就足够了。
如果你想要的效果是,对于不同的异常我们需要定制不同的处理逻辑,那就需要用到多分支了。

(3)异常的其他机构

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)主动触发异常

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 条件

上面的使用if条件实现主动触发异常,这里使用断言就可以实现相同的结果
ip_list=[
    # '1.1.1.1:8080',
    # '1.1.1.2:8081',
    # '1.1.1.3:8082',
]
assert len(ip_list) > 0    #当列表不为空时,正常执行下面的代码,否则报异常


(6)自定义异常

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处理异常