欢迎来到专栏《Python进阶》。在这个专栏中,我们会讲述Python的各种进阶操作,包括Python对文件、数据的处理,Python各种好用的库如NumPy、Scipy、Matplotlib、Pandas的使用等等。我们的初心就是带大家更好的掌握Python这门语言,让它能为我所用。

今天是《Python进阶》专栏的第四期,在本期中,我们将主要介绍Python中的异常与错误处理。

作者&编辑 | 汤兴旺

相信大家平时写程序时经常会碰到各种各样的异常和错误,而且一碰到程序报错时,心情就不好了。今天带大家学习下Python中的异常与错误处理。

1 异常与错误处理基本概念

在Python中,异常处理实际上就是当Python解释器检测到错误,触发异常时,程序员事先编写特定的代码会起作用,这时它会来捕捉这个异常(这段代码与程序逻辑无关,与异常处理有关),如果捕捉成功则进入另外一个处理分支,执行你为其定制的逻辑,使程序不会崩溃。简单来说,异常处理实际上就是防止程序崩溃用的。



python 存储错误信息 python处理错误_python异常处理

之所以要在程序中加入适当的异常处理,是有理由的。相信你有过这样的经验,你使用Python解释器去执行程序时,当程序运行到某处时,程序突然报错,实际上这就是Python解释器检测到了一个错误,触发异常,异常触发后且没被处理的情况下,程序就在当前异常处终止,后面的代码不会运行。因此异常处理非常重要,良好的异常处理机制可以帮助您增强程序的健壮性与容错性。

2 常见的异常与错误种类

Python中不同的异常可以用不同的类型去标识,不同的类对象标识不同的异常,一个异常标识一种错误。

(1)  AttributeError:试图访问一个对象没有的属性,比如foo.y,但是foo没有属性y

(2)  IOError:输入/输出异常;基本上是无法打开文件 

(3)  ImportError:无法引入模块或包;基本上是路径问题或名称错误 (4)  IndentationError:语法错误(的子类);代码没有正确对齐 

(5)  IndexError:下标索引超出序列边界

(6)  KeyError:试图访问字典里不存在的键 

(7)  KeyBoardInterrupt:ctrl+c被按下(在cmd命令下)

(8)  NameError:使用一个还未被赋予对象的变量 

(9)  SyntaxError:Python代码非法错误。代码不能编译

(10) TypeError:传入对象类型与要求的不符合

(11) UnboundLocalError:试图访问一个还未被设置的局部变量,基本上是由于另有一个同名的全局变量,才导致你以为正在访问它

(12) ValueError:传入一个调用者不期望的值,即使值的类型是正确的

上面就是Python中常见的一些内置异常,在Python中,异常也是对象,另外BaseException是所有内置异常的基类,但用户定义的类并不直接继承BaseException,所有的异常类都是从Exception继承,且都在exceptions模块中定义。下面我们就来创建一个自定义异常类Error类,基类为Exception,用于在异常触发时输出更多的信息。

class Error(Exception):
def __init__(self, msg):
     self.msg = msg
def __str__(self):
     return self.msg
try:
     raise Error('类型错误')
except Error as e:
    print('错误你又来了', e.msg)

上面的代码中,Error继承自基类Exception。另外在在try语句块中,抛出用户自定义的异常后执行except部分,变量 e 是用于创建MyError类的实例。

在上面的介绍中,我们提及了try...except...结构,实际上它是python的异常捕获常用的语法结构,下面我们来详细说明这个语法。

3 异常捕获

在Python中,每一种异常都有一个类型,也会有一种特定的语法结构用来进行异常处理。下面是异常处理结构的基本语法:

try:

   被检测的代码块

except 异常类型:

    try中一旦检测异常,就执行except中的代码块

现在我们用上面这个基本语法写个简单的代码,如下:

try:
    f = open("test.txt", 'r')
    f.write("这是一个测试文件")
except IOError as e:
    print('异常:',e)

输出:



python 存储错误信息 python处理错误_python 存储错误信息_02

除了上面最基本的语法结构外,还有许多其他衍生出来的结构。如下:

3.1 捕获所有异常

try:

except:

    print('异常说明')

该方法不需要指定异常类型,它会捕获所有可能会出现的异常,包括键盘中断和程序退出请求。此时用用sys.exit()就无法退出程序了,因为它属于异常。所以捕获异常建议大家慎用,它有非常大的弊端。

3.2 捕获多个异常

捕获多个异常有两种方式,第一种是一个except同时处理多个异常,不区分优先级:

try:

    < 语句 >

except (< 异常名1 >, < 异常名2 >, ...):

    print('异常说明')

第二种是区分优先级的:

try:

except :

      print('异常说明1')

except :

      print('异常说明2')

except :

      print('异常说明3')

except :

      print('异常说明4')

捕获多种异常通常按照下面的规则来处理:

执行try下的语句,如果引发异常,则执行过程会跳到第一个except语句。如果第一个except中定义的异常与引发的异常匹配,则执行该except中的语句。如果引发的异常不匹配第一个except,则会搜索第二个except。如果所有的except都不匹配,则异常会传递到下一个调用本代码的最高层try代码中。

话不多说,我们来看下下面的示例:

try:

d = dict(a=1, b=2)
    print(d['f'])  # KeyError
    print(a)  # NameError
except (KeyError, NameError) as e:
    print(e)

else:

 

print("如果没有捕获到异常, 执行else语句")
finally:
    print("有异常或者没有异常都会执行")



python 存储错误信息 python处理错误_python 存储错误信息_03

可以看到执行了异常错误KeyError, NameError。

在上面的语句中我还引出了else语句和finally语句。其中else语句的用法是如果判断完没有某些异常之后还想做其它事,就可以使用else语句。而finally语句与else语句正好相反,它是指无论是否发生异常都将会执行最后的代码。

3.3抛出异常

通常我们用raise:关键字来抛出异常,其中raise后面紧跟着抛出异常的名称和抛出异常的详细信息。 

class numError(BaseException):
    pass
def get_num(num):
    if 0 < num <= 300:
        print(num)
    else:
        # 如果raise抛出异常, 程序后面的代码不会再执行.
     raise  numError('invaild num')
get_num(1000)
get_num(100)



python 存储错误信息 python处理错误_python 存储错误信息_04

总结

本期我们介绍了如何使用Python中的异常与错误处理,希望今后您的程序不再有错误。

下期预告:Python库matplotlib的高级应用

有三AI编程与开源框架