8. 错误和异常

    语法错误又称解析错误,可能是你在学习Python 时最容易遇到的错误:

    解析器会输出出现语法错误的那一行,并显示一个“箭头”,指向这行里面检测到的第一个错误。 错误是由箭头指示的位置 上面 的 token 引起的(或者至少是在这里被检测出的):在示例中,在 print() 这个函数中检测到了错误,因为在它前面少了个冒号 () 。文件名和行号也会被输出,以便输入来自脚本文件时你能知道去哪检查。

    8.2. 异常

    即使语句或表达式在语法上是正确的,但在尝试执行时,它仍可能会引发错误。 在执行时检测到的错误被称为 异常,异常不一定会导致严重后果:你将很快学会如何在 Python 程序中处理它们。 但是,大多数异常并不会被程序处理,此时会显示如下所示的错误信息:

    1. >>> 10 * (1/0)
    2. Traceback (most recent call last):
    3. File "<stdin>", line 1, in <module>
    4. ZeroDivisionError: division by zero
    5. >>> 4 + spam*3
    6. Traceback (most recent call last):
    7. File "<stdin>", line 1, in <module>
    8. NameError: name 'spam' is not defined
    9. >>> '2' + 2
    10. Traceback (most recent call last):
    11. File "<stdin>", line 1, in <module>
    12. TypeError: Can't convert 'int' object to str implicitly

    错误信息的最后一行告诉我们程序遇到了什么类型的错误。异常有不同的类型,而其类型名称将会作为错误信息的一部分中打印出来:上述示例中的异常类型依次是:ZeroDivisionError, 和 TypeError。作为异常类型打印的字符串是发生的内置异常的名称。对于所有内置异常都是如此,但对于用户定义的异常则不一定如此(虽然这是一个有用的规范)。标准的异常类型是内置的标识符(而不是保留关键字)。

    这一行的剩下的部分根据异常类型及其原因提供详细信息。

    错误信息的前一部分以堆栈回溯的形式显示发生异常时的上下文。通常它包含列出源代码行的堆栈回溯;但是它不会显示从标准输入中读取的行。

    列出了内置异常和它们的含义。

    可以编写处理所选异常的程序。请看下面的例子,它会要求用户一直输入,直到输入的是一个有效的整数,但允许用户中断程序(使用 Control-C 或操作系统支持的其他操作);请注意用户引起的中断可以通过引发 KeyboardInterrupt 异常来指示。:

    1. >>> while True:
    2. ... try:
    3. ... x = int(input("Please enter a number: "))
    4. ... break
    5. ... except ValueError:
    6. ... print("Oops! That was no valid number. Try again...")
    7. ...

    语句的工作原理如下:

    • 首先,执行 try 子句try 和 关键字之间的(多行)语句)。

    • 如果没有异常发生,则跳过 except 子句 并完成 try 语句的执行。

    • 如果发生的异常和 except 子句中指定的异常不匹配,则将其传递到外部的 语句中;如果没有找到处理程序,则它是一个 未处理异常,执行将停止并显示如上所示的消息。

    A try statement may have more than one except clause, to specify handlers for different exceptions. At most one handler will be executed. Handlers only handle exceptions that occur in the corresponding try clause, not in other handlers of the same statement. An except clause may name multiple exceptions as a parenthesized tuple, for example:

    1. ... except (RuntimeError, TypeError, NameError):
    2. ... pass

    如果发生的异常和 except 子句中的类是同一个类或者是它的基类,则异常和 except 子句中的类是兼容的(但反过来则不成立 — 列出派生类的 except 子句与基类不兼容)。 例如,下面的代码将依次打印 B, C, D

    1. class B(Exception):
    2. pass
    3. class C(B):
    4. pass
    5. class D(C):
    6. pass
    7. for cls in [B, C, D]:
    8. try:
    9. raise cls()
    10. except D:
    11. print("D")
    12. except C:
    13. print("C")
    14. except B:
    15. print("B")

    请注意如果 except 子句被颠倒(把 except B 放到第一个),它将打印 B,B,B — 即第一个匹配的 except 子句被触发。

    最后的 except 子句可以省略异常名,以用作通配符。但请谨慎使用,因为以这种方式很容易掩盖真正的编程错误!它还可用于打印错误消息,然后重新引发异常(同样允许调用者处理异常):

    except 语句有一个可选的 else 子句,在使用时必须放在所有的 except 子句后面。对于在 try 子句不引发异常时必须执行的代码来说很有用。 例如:

    1. for arg in sys.argv[1:]:
    2. try:
    3. f = open(arg, 'r')
    4. print('cannot open', arg)
    5. else:
    6. print(arg, 'has', len(f.readlines()), 'lines')
    7. f.close()

    The use of the clause is better than adding additional code to the try clause because it avoids accidentally catching an exception that wasn’t raised by the code being protected by the … except statement.

    发生异常时,它可能具有关联值,也称为异常 参数 。参数的存在和类型取决于异常类型。

    except 子句可以在异常名称后面指定一个变量。这个变量和一个异常实例绑定,它的参数存储在 instance.args 中。为了方便起见,异常实例定义了 ,因此可以直接打印参数而无需引用 .args 。也可以在抛出之前首先实例化异常,并根据需要向其添加任何属性。:

    1. ... raise Exception('spam', 'eggs')
    2. ... except Exception as inst:
    3. ... print(type(inst)) # the exception instance
    4. ... print(inst.args) # arguments stored in .args
    5. ... print(inst) # __str__ allows args to be printed directly,
    6. ... # but may be overridden in exception subclasses
    7. ... x, y = inst.args # unpack args
    8. ... print('x =', x)
    9. ... print('y =', y)
    10. ...
    11. <class 'Exception'>
    12. ('spam', 'eggs')
    13. ('spam', 'eggs')
    14. x = spam
    15. y = eggs

    如果异常有参数,则它们将作为未处理异常的消息的最后一部分(’详细信息’)打印。

    异常处理程序不仅处理 try 子句中遇到的异常,还处理 try 子句中调用(即使是间接地)的函数内部发生的异常。例如:

    1. >>> def this_fails():
    2. ... x = 1/0
    3. ...
    4. >>> try:
    5. ... this_fails()
    6. ... except ZeroDivisionError as err:
    7. ... print('Handling run-time error:', err)
    8. ...
    9. Handling run-time error: division by zero

    8.4. 抛出异常

    1. >>> raise NameError('HiThere')
    2. Traceback (most recent call last):
    3. File "<stdin>", line 1, in <module>
    4. NameError: HiThere

    唯一的参数就是要抛出的异常。这个参数必须是一个异常实例或者是一个异常类(派生自 Exception 的类)。如果传递的是一个异常类,它将通过调用没有参数的构造函数来隐式实例化:

    如果你需要确定是否引发了异常但不打算处理它,则可以使用更简单的 语句形式重新引发异常

    1. >>> try:
    2. ... raise NameError('HiThere')
    3. ... except NameError:
    4. ... print('An exception flew by!')
    5. ... raise
    6. ...
    7. An exception flew by!
    8. Traceback (most recent call last):
    9. File "<stdin>", line 2, in <module>
    10. NameError: HiThere

    程序可以通过创建新的异常类来命名它们自己的异常(有关Python 类的更多信息,请参阅 )。异常通常应该直接或间接地从 类派生。

    可以定义异常类,它可以执行任何其他类可以执行的任何操作,但通常保持简单,只提供一些属性,这些属性允许处理程序为异常提取有关错误的信息。 在创建可能引发多个不同错误的模块时,通常的做法是为该模块定义的异常创建基类,并为不同错误条件创建特定异常类的子类:

    1. class Error(Exception):
    2. """Base class for exceptions in this module."""
    3. pass
    4. class InputError(Error):
    5. """Exception raised for errors in the input.
    6. Attributes:
    7. expression -- input expression in which the error occurred
    8. message -- explanation of the error
    9. self.expression = expression
    10. self.message = message
    11. class TransitionError(Error):
    12. """Raised when an operation attempts a state transition that's not
    13. allowed.
    14. Attributes:
    15. previous -- state at beginning of transition
    16. next -- attempted new state
    17. message -- explanation of why the specific transition is not allowed
    18. """
    19. def __init__(self, previous, next, message):
    20. self.previous = previous
    21. self.next = next
    22. self.message = message

    大多数异常都定义为名称以“Error”结尾,类似于标准异常的命名。

    许多标准模块定义了它们自己的异常,以报告它们定义的函数中可能出现的错误。有关类的更多信息,请参见 章节。

    8.6. 定义清理操作

    try 语句有另一个可选子句,用于定义必须在所有情况下执行的清理操作。例如:

    1. >>> try:
    2. ... raise KeyboardInterrupt
    3. ... finally:
    4. ... print('Goodbye, world!')
    5. ...
    6. Goodbye, world!
    7. KeyboardInterrupt
    8. Traceback (most recent call last):
    9. File "<stdin>", line 2, in <module>

    A finally clause is always executed before leaving the statement, whether an exception has occurred or not. When an exception has occurred in the try clause and has not been handled by an clause (or it has occurred in an except or clause), it is re-raised after the finally clause has been executed. The clause is also executed “on the way out” when any other clause of the try statement is left via a , continue or statement. A more complicated example:

    1. >>> def divide(x, y):
    2. ... try:
    3. ... result = x / y
    4. ... except ZeroDivisionError:
    5. ... print("division by zero!")
    6. ... else:
    7. ... print("result is", result)
    8. ... finally:
    9. ... print("executing finally clause")
    10. ...
    11. >>> divide(2, 1)
    12. result is 2.0
    13. executing finally clause
    14. >>> divide(2, 0)
    15. division by zero!
    16. executing finally clause
    17. >>> divide("2", "1")
    18. executing finally clause
    19. Traceback (most recent call last):
    20. File "<stdin>", line 1, in <module>
    21. File "<stdin>", line 3, in divide
    22. TypeError: unsupported operand type(s) for /: 'str' and 'str'

    As you can see, the finally clause is executed in any event. The raised by dividing two strings is not handled by the except clause and therefore re-raised after the clause has been executed.

    在实际应用程序中,finally 子句对于释放外部资源(例如文件或者网络连接)非常有用,无论是否成功使用资源。

    某些对象定义了在不再需要该对象时要执行的标准清理操作,无论使用该对象的操作是成功还是失败,清理操作都会被执行。 请查看下面的示例,它尝试打开一个文件并把其内容打印到屏幕上。:

    1. with open("myfile.txt") as f:
    2. for line in f:

    执行完语句后,即使在处理行时遇到问题,文件 f 也始终会被关闭。和文件一样,提供预定义清理操作的对象将在其文档中指出这一点。