Python教程:一文了解Python的异常处理知识

简介: 在编程中,异常(exception)是指在程序执行过程中出现的错误或意外情况。当发生异常时,程序会中断执行,并将控制权转移到异常处理代码,以便对异常进行捕获和处理。

1.异常处理的基础知识

1.1 什么是异常?

在编程中,异常(exception)是指在程序执行过程中出现的错误或意外情况。当发生异常时,程序会中断执行,并将控制权转移到异常处理代码,以便对异常进行捕获和处理。

异常可以分为两类:内建异常(built-in exceptions)和自定义异常(custom exceptions)。内建异常是 Python 提供的一组预定义异常类型,而自定义异常是根据特定需求创建的用户自定义的异常类型。

1.2 异常处理的目的和好处

异常处理的目的是在程序出错时提供适当的应对措施,以确保程序的稳定性和可靠性。通过合理处理异常,我们可以:

  • 防止程序崩溃,提高程序的容错能力。
  • 提供友好的错误信息,方便调试和排查问题。
  • 实现错误恢复和资源释放,避免资源泄漏。

1.3 异常处理的基本语法(try-except 块)

在 Python 中,使用 try-except 块来处理异常。try 块用于包含可能引发异常的代码,而 except 块用于捕获和处理异常。

下面是 try-except 块的基本语法:

try:
    # 可能引发异常的代码
except ExceptionType:
    # 处理特定类型的异常
except AnotherExceptionType:
    # 处理其他类型的异常
...
else:
    # 如果没有发生任何异常,执行该代码块
finally:
    # 无论是否发生异常,最终都会执行该代码块

image.gif

try 块中,我们放置可能引发异常的代码。如果发生了异常,并且异常类型匹配了某个 except 子句中指定的异常类型,那么相应的处理代码将被执行。可以有多个 except 子句来处理不同类型的异常。

else 块是可选的,用于在没有发生任何异常时执行特定的代码块。finally 块也是可选的,无论是否发生异常,都会执行其中的代码块。

2.处理常见异常类型

Python 提供了一组内建的异常类型,如 NameErrorTypeErrorValueError 等。当程序出现对应类型的异常时,我们可以捕获并处理这些异常。

2.1 NameError 异常

NameError 异常表示使用了一个未定义的变量或函数名。下面是一个示例:

try:
    print(x)  # x 未定义
except NameError:
    print("变量 x 未定义")

image.gif

2.2 TypeError 异常

TypeError 异常表示使用了错误类型的对象或进行了不兼容的操作。下面是一个示例:

try:
    x = '5'
    y = 2
    z = x + y  # 字符串与整数相加
except TypeError:
    print("类型错误:无法将字符串和整数相加")

image.gif

2.3 ValueError 异常

ValueError 异常表示使用了具有正确类型但具有无效值的对象。下面是一个示例:

try:
    x = int('abc')  # 无法将字符串 'abc' 转换为整数
except ValueError:
    print("值错误:无法将字符串转换为整数")

image.gif

处理其他内建异常的方式与上述示例类似。

2.4 Python 中常见的内建异常类型列表:

  1. Exception:所有异常的基类
  2. StopIteration:迭代器没有更多的值
  3. StopAsyncIteration:异步迭代器没有更多的值
  4. ArithmeticError:所有数值计算错误的基类
  5. FloatingPointError:浮点计算错误
  6. OverflowError:数值运算超出最大限制
  7. ZeroDivisionError:除数为零
  8. AssertionErrorassert 语句失败
  9. AttributeError:对象没有这样的属性
  10. EOFError:没有内建输入,到达文件末尾
  11. ImportError:导入模块失败
  12. ModuleNotFoundError:找不到模块
  13. LookupError:无效数据查找的基类
  • IndexError:索引超出序列范围
  • KeyError:字典中的键不存在
  1. NameError:未声明/初始化的变量名
  2. UnboundLocalError:访问未初始化的本地变量
  3. OSError:操作系统产生的错误
  4. FileNotFoundError:文件不存在
  5. PermissionError:没有足够的权限
  6. TypeError:不同类型间的无效操作
  7. ValueError:传递给函数的参数类型正确但值不合适
  8. RuntimeError:检测到运行时错误
  9. NotImplementedError:抽象方法在子类中没有实现
  10. RecursionError:递归调用层次太深

3.处理多个异常

try-except 块中,可以使用多个 except 子句来处理不同类型的异常。下面是一个示例:

try:
    # 可能引发异常的代码
except ExceptionType1:
    # 处理 ExceptionType1 异常
except ExceptionType2:
    # 处理 ExceptionType2 异常

image.gif

在处理多个异常时,需要注意以下几点:

  • 异常类型应按照从具体到一般的顺序排列,以免某个异常被通用异常类型的 except 子句捕获而无法执行特定的处理代码。
  • 可以在一个 except 子句中捕获多个异常类型,例如 except (ExceptionType1, ExceptionType2):
  • 也可以使用多个 except 子句来处理相同的异常类型,以便在不同的子句中实现不同的处理逻辑。

下面是处理多个异常的示例:

try:
    x = 5 / 0
    print(x)
except ZeroDivisionError:
    print("除数不能为零")
except TypeError:
    print("类型错误")

image.gif

4.使用 elsefinally

除了 tryexcept 块之外,还可以使用 elsefinally 块来进一步处理异常。

4.1 else

else 块用于在没有发生任何异常时执行一些特定的代码。例如,在读取文件时,如果没有发生异常,则可以在 else 块中处理文件数据。

下面是一个示例:

try:
    file = open("data.txt", "r")
except FileNotFoundError:
    print("文件未找到")
else:
    data = file.read()
    print(data)
    file.close()

image.gif

在上述示例中,如果成功打开文件并读取数据,那么 else 块将被执行。

4.2 finally

finally 块用于定义无论是否发生异常都必须执行的代码块。一般情况下,我们可以在 finally 块中进行资源的释放和清理操作,如关闭文件、释放数据库连接等。

下面是一个示例:

try:
    file = open("data.txt", "r")
except FileNotFoundError:
    print("文件未找到")
finally:
    file.close()  # 无论是否发生异常,都会执行该代码块来关闭文件

image.gif

在上述示例中,无论是否发生异常,file.close() 都会被执行来关闭文件。

5.自定义异常

除了内建的异常类型之外,我们还可以自定义异常类来满足特定的需求。自定义异常类需要继承自 Exception 类或其子类,并可以添加额外的属性和方法。

下面是一个自定义异常类的示例:

class CustomError(Exception):
    def __init__(self, message):
        self.message = message
try:
    raise CustomError("自定义异常信息")
except CustomError as e:
    print(e.message)

image.gif

在上述示例中,我们定义了一个名为 CustomError 的自定义异常类,它继承自 Exception 类。然后,我们通过 raise 关键字抛出了一个自定义异常对象,并在 except 块中捕获并打印了异常信息。

6.异常处理的最佳实践

在处理异常时,有一些最佳实践可以帮助我们编写更清晰、可靠的代码。

6.1 适度使用异常

异常处理是用来处理异常情况的,而不应该用来控制程序流程。因此,我们应该避免过度使用异常,将其限制在真正的异常情况下。

6.2 具体异常优于通用异常

尽可能使用具体的异常类型来捕获和处理异常,而不是使用通用的异常类型。这样可以更精确地定位问题,并提供更有针对性的处理逻辑。

6.3 不要忽略异常

避免使用空的 except 子句来完全忽略异常。这样可能会导致问题被掩盖,难以调试和定位。

6.4 清理资源和关闭文件

finally 块中进行资源的释放和清理操作,以确保无论是否发生异常,都能正确地释放资源,避免资源泄漏。

7.高级异常处理技术

除了基本的 try-except 块之外,Python 还提供了一些高级异常处理技术。

7.1 使用 with 语句

with 语句是一种上下文管理器,它可以自动管理资源的分配和释放。通过使用 with 语句,我们可以简化资源管理的代码,并确保在退出代码块时正确地释放资源。

下面是一个读取文件的示例:

with open("data.txt", "r") as file:
    data = file.read()
    print(data)

image.gif

在上述示例中,open() 函数返回一个文件对象,然后将其赋值给变量 file。在 with 代码块中,我们可以自由地使用 file 对象,而不必担心在退出代码块时忘记关闭文件。当代码块执行完毕时,with 语句会自动关闭文件。

7.2 控制特定代码块中的异常

有时候,我们只想在特定的代码块中捕获和处理异常,而不影响其他代码块。Python 提供了 try-except 块的嵌套使用来实现这一目标。

下面是一个示例:

try:
    # 其他代码
    try:
        # 特定代码块
    except SpecificException:
        # 处理 SpecialException 异常
    # 其他代码
except GeneralException:
    # 处理 GeneralException 异常

image.gif

在上述示例中,外层的 try-except 块用于处理通用的异常类型,而内层的 try-except 块用于处理特定的异常类型。这样可以实现对特定代码块中的异常进行控制。

7.3 调试和记录异常

在开发过程中,调试和记录异常是非常重要的。它们可以帮助我们分析和解决异常,并提供有关

异常发生的有用信息。

7.4 调试异常信息

在开发过程中,我们可以通过输出异常信息来调试程序。Python 的内建模块 traceback 提供了异常堆栈跟踪信息,可以帮助我们定位异常发生的位置。

下面是一个示例:

import traceback
try:
    x = 5 / 0
except ZeroDivisionError as e:
    traceback.print_exc()

image.gif

在上述示例中,我们导入了 traceback 模块,并在捕获 ZeroDivisionError 异常时使用 traceback.print_exc() 打印异常堆栈跟踪信息。这样可以帮助我们更准确地了解异常的起因和位置。

7.5 记录异常信息

除了直接在控制台输出异常信息外,我们还可以将异常信息写入日志文件或数据库,以便后续分析和排查问题。

下面是一个示例:

import logging
try:
    x = 5 / 0
except ZeroDivisionError as e:
    logging.basicConfig(filename='error.log', level=logging.ERROR)
    logging.error(f'An error occurred: {e}')

image.gif

在上述示例中,我们使用 Python 的 logging 模块将异常信息记录到名为 error.log 的日志文件中。通过记录异常信息,我们可以在生产环境中更好地跟踪和解决问题。

8.异常处理常见面试题

面试题:介绍一下 Python 中 try-except-else-finally 的使用方法,并举例说明。

答案:

  • try-except 用来捕获可能发生异常的代码块。
  • elsetry 块没有引发异常时执行。
  • finally 始终执行,无论是否发生异常。

代码示例:

def divide_numbers(a, b):
    try:
        result = a / b
    except ZeroDivisionError:
        print("Error: Division by zero!")
    else:
        print(f"Division result: {result}")
    finally:
        print("Execution complete.")
divide_numbers(10, 2)
divide_numbers(10, 0)

image.gif

输出结果:

Division result: 5.0

Execution complete.

Error: Division by zero!

Execution complete.

在上面的示例中,我们定义了一个函数 divide_numbers,尝试对两个数进行除法运算。第一次调用传入参数为 (10, 2) 没有引发异常,所以会执行 else 块和 finally 块;第二次调用传入参数为 (10, 0) 引发了 ZeroDivisionError,会执行 except 块和 finally 块。

目录
相关文章
|
8天前
|
存储 Python
SciPy 教程 之 SciPy 稀疏矩阵 4
SciPy 教程之 SciPy 稀疏矩阵 4:介绍稀疏矩阵的概念、类型及其在科学计算中的应用。SciPy 的 `scipy.sparse` 模块提供了处理稀疏矩阵的工具,重点讲解了 CSC 和 CSR 两种格式,并通过示例演示了如何创建和操作 CSR 矩阵。
31 3
|
2天前
|
机器学习/深度学习 数据处理 Python
SciPy 教程 之 SciPy 空间数据 4
本教程介绍了SciPy的空间数据处理功能,主要通过scipy.spatial模块实现。内容涵盖空间数据的基本概念、距离矩阵的定义及其在生物信息学中的应用,以及如何计算欧几里得距离。示例代码展示了如何使用SciPy计算两点间的欧几里得距离。
16 5
|
1天前
|
机器学习/深度学习 Python
SciPy 教程 之 SciPy 空间数据 6
本教程介绍了SciPy处理空间数据的方法,包括使用scipy.spatial模块进行点位置判断、最近点计算等内容。还详细讲解了距离矩阵的概念及其应用,如在生物信息学中表示蛋白质结构等。最后,通过实例演示了如何计算两点间的余弦距离。
9 3
|
4天前
|
Python
SciPy 教程 之 SciPy 图结构 7
《SciPy 教程 之 SciPy 图结构 7》介绍了 SciPy 中处理图结构的方法。图是由节点和边组成的集合,用于表示对象及其之间的关系。scipy.sparse.csgraph 模块提供了多种图处理功能,如 `breadth_first_order()` 方法可按广度优先顺序遍历图。示例代码展示了如何使用该方法从给定的邻接矩阵中获取广度优先遍历的顺序。
14 2
|
5天前
|
算法 Python
SciPy 教程 之 SciPy 图结构 5
SciPy 图结构教程,介绍图的基本概念和SciPy中处理图结构的模块scipy.sparse.csgraph。重点讲解贝尔曼-福特算法,用于求解任意两点间最短路径,支持有向图和负权边。通过示例演示如何使用bellman_ford()方法计算最短路径。
14 3
|
5天前
|
缓存 测试技术 Apache
告别卡顿!Python性能测试实战教程,JMeter&Locust带你秒懂性能优化💡
告别卡顿!Python性能测试实战教程,JMeter&Locust带你秒懂性能优化💡
14 1
|
6天前
|
算法 索引 Python
SciPy 教程 之 SciPy 图结构 3
SciPy 图结构教程:介绍图的基本概念、节点和边的定义,以及如何使用 SciPy 的 `scipy.sparse.csgraph` 模块处理图结构。重点讲解 Dijkstra 最短路径算法及其在 SciPy 中的应用,包括 `dijkstra()` 方法的参数设置和使用示例。
10 0
|
6天前
|
Python
SciPy 教程 之 SciPy 图结构 2
《SciPy 教程 之 SciPy 图结构 2》介绍了图结构作为算法学中的重要框架,通过 `scipy.sparse.csgraph` 模块处理图结构。文章示例展示了如何使用 `connected_components()` 方法查找所有连接组件,通过创建稀疏矩阵并调用该方法实现。
8 0
|
7天前
|
算法 Python
SciPy 教程 之 SciPy 图结构 1
SciPy 图结构教程介绍了图的基本概念及其在算法中的应用。图由节点和边组成,节点代表对象,边表示对象间的连接。SciPy 的 `scipy.sparse.csgraph` 模块提供了处理图结构的工具。邻接矩阵用于表示节点间的连接关系,分为有向图和无向图两种类型。无向图的边是双向的,而有向图的边则有明确的方向。
17 0
|
7天前
|
存储 Python
SciPy 教程 之 SciPy 稀疏矩阵 5
SciPy 稀疏矩阵教程介绍了稀疏矩阵的概念及其在科学计算中的应用。SciPy 的 `scipy.sparse` 模块提供了处理稀疏矩阵的功能,主要使用 CSC(压缩稀疏列)和 CSR(压缩稀疏行)两种格式。通过示例展示了如何创建 CSR 矩阵、查看非零元素及转换为 CSC 格式。
20 0
下一篇
无影云桌面