热门搜索 :
考研考公
您的当前位置:首页正文

Python 的异常处理

来源:东饰资讯网

在执行程序的过程中,可能会遇到多多少少的“意外情况”,比如除数为 0,文件找不到,变量未声明等。解释器在发现这些异常错误后,会当机立断终止程序的运行,如果我们想程序继续运行,提高代码的健壮性,就需要用到异常处理。

try 和 except

Python 中使用 try 关键字来捕获异常,使用 except 关键字来处理异常,没有 catch 关键字。
不进行异常处理的情况:

def devide(a,b):
    return a/b
        
devide(1,0)

执行 devide 函数,程序直接挂掉了,抛出一个 ZeroDivisionError 的异常:

Traceback (most recent call last):
  File "C:\Users\Charley\Desktop\py\py.py", line 4, in <module>
    devide(1,0)
  File "C:\Users\Charley\Desktop\py\py.py", line 2, in devide
    return a/b
ZeroDivisionError: division by zero
[Finished in 0.6s with exit code 1]
[shell_cmd: python -u "C:\Users\Charley\Desktop\py\py.py"]
[dir: C:\Users\Charley\Desktop\py]
[path: C:\Python27\;C:\Python27\Scripts;C:\WINDOWS\system32;C:\WINDOWS;C:\WINDOWS\System32\Wbem;C:\WINDOWS\System32\WindowsPowerShell\v1.0\;D:\nvm;D:\nodejs;C:\Program Files (x86)\Windows Kits\8.1\Windows Performance Toolkit\;C:\Program Files\Microsoft SQL Server\110\Tools\Binn\;C:\ProgramData\chocolatey\bin;C:\Program Files (x86)\GtkSharp\2.12\bin;C:\Program Files\MySQL\MySQL Utilities 1.6\;C:\Users\Charley\AppData\Local\Programs\Python\Python35-32\Scripts\;C:\Users\Charley\AppData\Local\Programs\Python\Python35-32\;D:\Git\bin;D:\Sublime Text 3;D:\MinGW\bin\;D:\MinGW\bin\;D:\Microsoft VS Code\bin;D:\Java\jdk 8.0\bin;D:\Android\sdk;C:\Program Files\MySQL\MySQL Server 5.7\bin;]

修改代码,对 ZeroDivisionError 异常进行捕获处理:

def devide(a,b):
    try:
        return a/b
    except ZeroDivisionError:
        print("除数不能为零!")
        
devide(1,0)

执行结果如下:

除数不能为零!

程序没有挂掉,并对相应的异常进行了处理。

捕获多个异常

我们还可以捕获多个异常:

def devide(a,b):
    try:
        print(num)
        return a/b
    except ZeroDivisionError:
        print("除数不能为零!")
    except NameError:
        print("变量不存在!")

devide(1,0)

运行结果:

变量不存在!

上面的异常处理可以捕获多个异常情况,如果触发了 ZeroDivisionError 异常,就输出 除数不能为零! 提示语,如果触发了 NameError 异常,就输出 变量不存在! 提示语。
在捕获多个异常时,也可以进行简写:

def devide(a,b):
    try:
        print(num)
        return a/b
    except (ZeroDivisionError,NameError):
        print("发生了一些异常情况!请检查代码")

devide(1,0)

Python2 中可以将异常直接使用逗号分隔,Python3 中必须将异常置于元组中。

异常类

def devide(a,b):
    try:
        return a/b
    except ZeroDivisionError:
        print(type(ZeroDivisionError))

devide(1,0)

执行结果:

<class 'type'>

这些异常都是类。所有的异常都有一个共同的父类 Exception,如果我们使用 Exception 来进行捕获,将捕获到所有的异常情况,使用 as 关键字可以查看具体的异常信息:

def devide(a,b):
    try:
        return a/b
    except Exception as res:
        print(res)

devide(1,0)

运行结果:

division by zero

raise 关键字

raise 关键字用来主动触发异常:

def devide(a,b):
    try:
        raise NameError
    except Exception:
        print("发生了一点错误!")

devide(1,1)

运行结果:

发生了一点错误!

上面我们手动触发了 NameError 异常,同样可以被捕获。

自定义异常

除了系统自带的异常之外,我们还可以对异常进行自定义,前面说到所有的异常都是类,因此我们也需要自定义异常类,该类应该以 Exception 类作为父类

class 没事儿就像搞点事情(Exception):
    def __init__(self,reason):
        self.reason = reason

def devide(a,b):
    try:
        raise 没事儿就像搞点事情("点事情是谁?")
    except Exception as res:
        print("发生了一点错误:%s"%res.reason)

devide(1,1)

运行结果如下:

发生了一点错误:点事情是谁?

新建对象是为了记录详细的异常信息,当然也可以直接抛出异常类:

class 没事儿就像搞点事情(Exception):
    def __init__(self,reason):
        self.reason = reason

def devide(a,b):
    try:
        raise 没事儿就像搞点事情
    except Exception as res:
        print("发生了一点错误")


devide(1,1)

运行结果如下:

发生了一点错误

else 关键字

如果没有出现异常,那么就会执行 else 中的代码:

def devide(a,b):
    try:
        print(a/b)
    except Exception as res:
        print("发生了一点错误:%s"%res)
    else:
        print("嘻嘻,没有发生错误哟!")

devide(1,1)

运行结果如下:

1.0
嘻嘻,没有发生错误哟!

finally 关键字

finally 是异常的出口,不管有没有异常,不管捕获了多少异常,都会执行 finally 中的语句:

def devide(a,b):
    try:
        print(a/b)
    except Exception as res:
        print("发生了一点错误:%s"%res)
    else:
        print("嘻嘻,没有发生错误哟!")
    finally:
        print("你若安好,便是晴天")

devide(1,1)

运行结果:

1.0
嘻嘻,没有发生错误哟!
你若安好,便是晴天

finally 中可以进行清理工作,比如关闭文件,该操作不论是否发生异常都应该被执行了,所以应该放在 finally 中。

异常传递

异常会在调用栈中逐层往外传递,直到被捕获到为止。内层函数产生了异常,如果自身没有进行处理,就将异常传递给调用它的函数,如果调用它的函数也不进行处理,再向外传递,直到传递给解释器,被解释器所捕获。

def a():
    print(num)

def b():
    a()

def c():
    try:
        b()
    except Exception as res:
        print(res)

c()

运行结果如下:

name 'num' is not defined

完。

Top