文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

Python 进阶:异常处理

2023-09-29 05:41

关注

异常是什么

Python 使用异常对象来表示异常状态,并在遇到错误时引发异常。异常对象未被处理(或捕获)时,程序将终止并显示一条错误消息(traceback)。

如果异常只能用来显示错误消息,就没多大意思了。但事实上,每个异常都是某个类的实例(这里是 ZeroDivisionError)。你能以各种方式引发和捕获这些实例,从而逮住错误并采取措施,而不是放任错误导致整个程序失败。

常见的异常

除零错误

1/0
Traceback (most recent call last):  File "", line 1, in ZeroDivisionError: integer division or modulo by zero

数组越界错误 

a = [1, 2, 3]print(a[3])
Traceback (most recent call last):  File "", line 1, in IndexError: list index out of range

运算符错误 

a = 123b = '123'print(a+b)
Traceback (most recent call last):  File "", line 1, in TypeError: unsupported operand type(s) for +: 'int' and 'str'

正如你看到的,出现问题时,将自动引发异常。 

捕捉异常 

try/except 语句

异常捕捉可以使用 try/except 语句。

try:    1/0except Exception as error:    print('发生如下错误:',error)
发生如下错误: division by zero

try 语句按照如下方式工作:

一个 try 语句可能包含多个 except 子句,分别来处理不同的特定的异常。最多只有一个分支会被执行。

处理程序将只针对对应的 try 子句中的异常进行处理,而不是其他的 try 的处理程序中的异常。

一个 except 子句可以同时处理多个异常,这些异常将被放在一个括号里成为一个元组,例如:

except (RuntimeError, TypeError, NameError):    pass

try/except...else 语句

try/except 语句还有一个可选的 else 子句,如果使用这个子句,那么必须放在所有的 except 子句之后。

else 子句将在 try 子句没有发生任何异常的时候执行。

以下实例在 try 语句中判断文件是否可以打开,如果打开文件时正常的没有发生异常则执行 else 部分的语句,读取文件内容:

# 正常打开try:    with open('solution.py', 'r') as f:        line = f.read()except:    print('出错了')else:    print('没有出错了')
没有出错了

try-finally 语句

try-finally 语句无论是否发生异常都将执行最后的代码。

以下实例中 finally 语句无论异常是否发生都会执行:

try:    1/1except:    print('出错了')else:    print('没有出错了')finally:    print('不管有没有出错,我100%会执行')
没有出错了不管有没有出错,我100%会执行

抛出异常 

首先要分清楚程序发生异常和程序执行错误,它们完全是两码事,程序由于错误导致的运行异常,是需要程序员想办法解决的;但还有一些异常,是程序正常运行的结果,比如用 raise 手动引发的异常。 

raise 语句

raise [exceptionName [(reason)]]

其中,用 [] 括起来的为可选参数,其作用是指定抛出的异常名称,以及异常信息的相关描述。如果可选参数全部省略,则 raise 会把当前错误原样抛出;如果仅省略 (reason),则在抛出异常时,将不附带任何的异常描述信息。

也就是说,raise 语句有如下三种常用的用法:

显然,每次执行 raise 语句,都只能引发一次执行的异常。首先,我们来测试一下以上 3 种 raise 的用法:

运行 rase

Traceback (most recent call last):File "/code/main.py", line 2, inimport solutionFile "/code/solution.py", line 1, inraiseRuntimeError: No active exception to reraise

运行 raise ZeroDivisionError

Traceback (most recent call last):File "/code/main.py", line 2, inimport solutionFile "/code/solution.py", line 3, inraise ZeroDivisionErrorZeroDivisionError

运行 raise ZeroDivisionError("除数不能为零")

Traceback (most recent call last):File "/code/main.py", line 2, inimport solutionFile "/code/solution.py", line 5, inraise ZeroDivisionError("除数不能为零")ZeroDivisionError: 除数不能为零

 当然,我们手动让程序引发异常,很多时候并不是为了让其崩溃。事实上,raise 语句引发的异常通常用 try except (else finally) 异常处理结构来捕获并进行处理。例如:

try:    a = 'hello'    if(not a.isdigit()):        raise ValueError("a 必须是数字")except ValueError as e:    print("引发异常:",repr(e))
引发异常: ValueError('a 必须是数字')

可以看到,当用字符串 a 不是数字时,程序会进入 if 判断语句,并执行 raise 引发 ValueError 异常。但由于其位于 try 块中,所以 raise 抛出的异常会被 try 捕获,并由 except 块进行处理。
因此,虽然程序中使用了 raise 语句引发异常,但程序的执行是正常的,手动抛出的异常并不会导致程序崩溃

正如前面所看到的,在使用 raise 语句时可以不带参数,例如:

try:    a = 'hello'    if(not a.isdigit()):        raise ValueError("a 必须是数字")except ValueError as e:    print("引发异常:",repr(e))    raise
Traceback (most recent call last):File "/code/main.py", line 2, inimport solutionFile "/code/solution.py", line 4, inraise ValueError("a 必须是数字")ValueError: a 必须是数字

这里重点关注位于 except 块中的 raise,由于在其之前我们已经手动引发了 ValueError 异常,因此这里当再使用 raise 语句时,它会再次引发一次。 

当在没有引发过异常的程序使用无参的 raise 语句时,它默认引发的是 RuntimeError 异常。例如:

try:    a = 'hello'    if(not a.isdigit()):        raiseexcept ValueError as e:    print("引发异常:",repr(e))
Traceback (most recent call last):File "/code/main.py", line 2, inimport solutionFile "/code/solution.py", line 4, inraiseRuntimeError: No active exception to reraise

内置异常 

在 Python 中,所有异常必须为一个派生自 BaseException 的类的实例。 在带有提及一个特定类的 except 子句的 try 语句中,该子句也会处理任何派生自该类的异常类(但不处理该子句本身所派生出的异常类)。通过子类化创建的两个不相关异常类永远是不等效的,既使它们具有相同的名称。

下面列出的内置异常可通过解释器或内置函数来生成。除非另有说明,它们都会具有一个提示导致错误详细原因的“关联值”。这可以是一个字符串或由多个信息项(例如一个错误码和一个解释错误的字符串)组成的元组。关联值通常会作为参数被传递给异常类的构造器。

用户代码可以引发内置异常。这可以被用于测试异常处理程序或报告错误条件,“就像” 在解释器引发了相同异常的情况时一样;但是请注意,没有任何机制能防止用户代码引发不适当的错误。

内置异常类可以被子类化以定义新的异常;鼓励程序员从 Exception 类或它的某个子类而不是从 BaseException 来派生新的异常。

当在 except 或 finally 子句中引发(或重新引发)异常时,__context__ 会被自动设为所捕获的最后一个异常;如果新的异常未被处理,则最终显示的回溯信息将包括原始的异常和最后的异常。

当引发一个新的异常(而不是简单地使用 raise 来重新引发当前在处理的异常)时,隐式的异常上下文可以通过使用带有 raise 的 from 来补充一个显式的原因:

raise new_exc from original_exc

跟在 from 之后的表达式必须为一个异常或 None。它将在所引发的异常上被设置为 __cause__。设置 __cause__ 还会隐式地将 __suppress_context__ 属性设为 True,这样使用 raise new_exc from None 可以有效地将旧异常替换为新异常来显示其目的 (例如将 KeyError 转换为 AttributeError),同时让旧异常在 __context__ 中保持可用状态以便在调试时进行内省。

除了异常本身的回溯以外,默认的回溯还会显示这些串连的异常。__cause__ 中的显式串连异常如果存在将总是显示。__context__ 中的隐式串连异常仅在 __cause__ 为 None 并且 __suppress_context__ 为假值时显示。

不论在哪种情况下,异常本身总会在任何串连异常之后显示,以便回溯的最后一行总是显示所引发的最后一个异常。

内置异常的类层级结构如下:

BaseException +-- SystemExit +-- KeyboardInterrupt +-- GeneratorExit +-- Exception      +-- StopIteration      +-- StopAsyncIteration      +-- ArithmeticError      |    +-- FloatingPointError      |    +-- OverflowError      |    +-- ZeroDivisionError      +-- AssertionError      +-- AttributeError      +-- BufferError      +-- EOFError      +-- ImportError      |    +-- ModuleNotFoundError      +-- LookupError      |    +-- IndexError      |    +-- KeyError      +-- MemoryError      +-- NameError      |    +-- UnboundLocalError      +-- OSError      |    +-- BlockingIOError      |    +-- ChildProcessError      |    +-- ConnectionError      |    |    +-- BrokenPipeError      |    |    +-- ConnectionAbortedError      |    |    +-- ConnectionRefusedError      |    |    +-- ConnectionResetError      |    +-- FileExistsError      |    +-- FileNotFoundError      |    +-- InterruptedError      |    +-- IsADirectoryError      |    +-- NotADirectoryError      |    +-- PermissionError      |    +-- ProcessLookupError      |    +-- TimeoutError      +-- ReferenceError      +-- RuntimeError      |    +-- NotImplementedError      |    +-- RecursionError      +-- SyntaxError      |    +-- IndentationError      |         +-- TabError      +-- SystemError      +-- TypeError      +-- ValueError      |    +-- UnicodeError      |         +-- UnicodeDecodeError      |         +-- UnicodeEncodeError      |         +-- UnicodeTranslateError      +-- Warning           +-- DeprecationWarning           +-- PendingDeprecationWarning           +-- RuntimeWarning           +-- SyntaxWarning           +-- UserWarning           +-- FutureWarning           +-- ImportWarning           +-- UnicodeWarning           +-- BytesWarning           +-- ResourceWarning

异常的基类

下列异常主要被用作其他异常的基类。

所有内置异常的基类。它不应该被用户自定义类直接继承(这种情况请使用 Exception)。如果在此类的实例上调用 str(),则会返回实例的参数表示,或者当没有参数时返回空字符串。args 传给异常构造器的参数元组。某些内置异常(例如 OSError)接受特定数量的参数并赋予此元组中的元素特殊的含义,而其他异常通常只接受一个给出错误信息的单独字符串。with_traceback(tb) 此方法将 tb 设为异常的新回溯信息并返回该异常对象。它通常以如下的形式在异常处理程序中使用:

try:    ...except SomeException:    tb = sys.exc_info()[2]    raise OtherException(...).with_traceback(tb)

所有内置的非系统退出类异常都派生自此类。所有用户自定义异常也应当派生自此类。

此基类用于派生针对各种算术类错误而引发的内置异常: OverflowErrorZeroDivisionErrorFloatingPointError

当与缓冲区相关的操作无法执行时将被引发。

此基类用于派生当映射或序列所使用的键或索引无效时引发的异常: IndexErrorKeyError。这可以通过 codecs.lookup() 来直接引发。

来源地址:https://blog.csdn.net/qq_55858843/article/details/127823492

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯