지금까지 오류 메시지에 대해 많이 언급하지 않았지만 몇 가지 예에서 사용되었습니다. Python에는 최소한 두 가지 유형의 오류가 있습니다. Syntax Errors 및 Exceptions
구문 분석 오류라고도 하는 구문 오류는 Python 초보자 사이에서 흔히 발생하는 불만 사항입니다.
>>> while True print('Hello world') File "<stdin>", line 1while True print('Hello world') ^SyntaxError: invalid syntax
Python 파서는 오류 줄을 반복적으로 인쇄하고 오류 줄에서 오류가 처음 감지된 위치를 가리키는 작은 화살표를 표시합니다. 오류는 화살표 앞의 문으로 인해 발생합니다(적어도 그것이 감지됩니다). 위 예에서는 콜론 :print()
함수에서 오류가 감지되었습니다. /코드>가 없습니다. 파일 이름과 줄 번호도 인쇄되어 프로그램이 스크립트에서 나올 때 문제를 더 쉽게 찾을 수 있습니다. print()
函数被侦测到,因为在它之前的冒号:
缺失了。文件名和行号也被打印出来,当程序来自脚本时可以方便定位问题。
即使语句或者表达式在语法上是正确的,但是在执行的时候也可能会发生错误。在执行期间侦测到的问题叫做异常,异常不是绝对的致命错误:接下来会介绍如何在Python中处理异常。然而大多数异常都不会被程序处理,它们最终会成为错误信息,展示如下:
>>> 10 * (1/0) Traceback (most recent call last): File "<stdin>", line 1, in <module>ZeroDivisionError: division by zero>>> 4 + spam*3Traceback (most recent call last): File "<stdin>", line 1, in <module>NameError: name 'spam' is not defined>>> '2' + 2Traceback (most recent call last): File "<stdin>", line 1, in <module>TypeError: Can't convert 'int' object to str implicitly
最后一行错误信息指出发生了什么。异常有不同类型,类型也作为错误信息的一部分打印出来:以上示例中的类型有除零异常ZeroDivisionError
, 名字异常NameError
以及类型异常TypeError
。作为异常类型打印的字符串是built-in异常的名字。对于所有built-in异常来说都是如此,尽管这个约定很有用,但是并不是所有用户自定义异常都会遵守。标准异常名字是built-in标识符(不是保留字)。
剩余行基于异常类型展示了详细的信息以及异常发生的原因。
前面部分错误信息以堆栈回溯的方式,展示异常发生的上下文信息。通常堆栈回溯中列出了源代码行;然而,当程序是从标准输入中读取时,不会展示行。
Built-in异常列举了所有built-in异常以及它们的意义。
Python程序允许处理指定异常。以下程序要求用户循环输入,直到输入为有效整数停止,但是也可以中断程序(使用Control-C
或者其他操作系统支持的手段);注意用户诱发的中断会抛出KeyboardInterrupt异常。
while True:try: x = int(input("Please enter a number: "))breakexcept ValueError:print("Oops! That was no valid number. Try again...")
try
语句的执行顺序为:
首先,执行在try
和except
之间的try子句
如果没有异常发生,跳过except子句,try
语句执行完毕
如果try子句执行期间有异常发生,该子句内剩余语句被跳过。接下来如果抛出的异常类型可以与关键字except
后的异常匹配,那么except子句执行,接着继续执行try
语句后的语句。
如果异常发生但是没有匹配到except后的异常,那么异常抛到外层try
语句;如果异常得不到处理,该异常成为未处理异常,导致程序终止并且像以上示例一样打印信息。
一个try
语句可以有多个except子句,为不同的异常指定处理方法。至多只有一个except子句会被执行。处理程序只会处理发生在对应try子句中的异常,而不会处理发生在同一try
语句中其他处理程序中的异常。一个except子句可以使用带括号的元组列出多个异常,就像这样:
... except (RuntimeError, TypeError, NameError): ... pass
except
子句中的类可以匹配类型是其子类或者它自己类型的异常(但反过来不行,except子句中的子类不会匹配父类异常)。例如以下代码会依次打印B, C, D:
class B(Exception):passclass C(B):passclass D(C):passfor cls in [B, C, D]:try:raise cls()except D:print("D")except C:print("C")except B:print("B")
注意以上的except子句若反过来写(except B
在前),那么将会打印B, B, B ——第一个except子句的匹配被触发。
最后一个except子句可以省略异常名字作为通配符。因为这样做会隐藏其他的真实程序错误,所有要谨慎使用。也可以用来打印错误信息并且重新抛出(允许调用者处理异常):
import systry: f = open('myfile.txt') s = f.readline() i = int(s.strip())except OSError as err:print("OS error: {0}".format(err))except ValueError:print("Could not convert data to an integer.")except:print("Unexpected error:", sys.exc_info()[0])raise
try
...except
语句有一个可选的else子句,该子句只能出现在所有except子句之后。若要求try子句没有抛出异常时必须执行一段代码,使用else子句很有用(译注:else
子句会被return, break, continue
for arg in sys.argv[1:]:try: f = open(arg, 'r')except OSError:print('cannot open', arg)else:print(arg, 'has', len(f.readlines()), 'lines') f.close()🎜🎜오류 메시지의 마지막 줄은 무슨 일이 일어났는지 나타냅니다. 다양한 유형의 예외가 있으며 해당 유형은 오류 메시지의 일부로도 인쇄됩니다. 위 예의 유형은 0으로 나누기 예외
ZeroDivisionError
, 이름 예외 NameError code> 및 예외 <code>TypeError
를 입력합니다. 예외 유형으로 인쇄된 문자열은 내장된 예외의 이름입니다. 이는 모든 내장 예외에 해당되며 이 규칙이 유용하더라도 모든 사용자 정의 예외가 이를 준수하는 것은 아닙니다. 표준 예외 이름은 내장 식별자(예약어 아님)입니다. 🎜🎜나머지 행에는 예외 유형과 예외가 발생한 이유에 따른 자세한 정보가 표시됩니다. 🎜🎜오류 메시지의 앞 부분은 예외의 컨텍스트 정보를 스택 추적 형식으로 표시합니다. 일반적으로 소스 코드 줄은 스택 추적에 나열되지만 프로그램이 표준 입력에서 읽을 때는 해당 줄이 표시되지 않습니다. 🎜🎜내장 예외에는 모든 내장 예외와 그 의미가 나열됩니다. 🎜🎜8.3 예외 처리🎜🎜Python 프로그램에서는 지정된 예외를 처리할 수 있습니다. 다음 프로그램에서는 입력이 유효한 정수가 될 때까지 사용자가 입력을 반복해야 하지만 프로그램을 중단할 수도 있습니다(Control-C
또는 운영 체제에서 지원하는 다른 수단 사용). 유도된 인터럽트는 KeyboardInterrupt 예외를 발생시킵니다. 🎜🎜>>> try: ... raise Exception('spam', 'eggs') ... except Exception as inst: ... print(type(inst)) # the exception instance... print(inst.args) # arguments stored in .args... print(inst) # __str__ allows args to be printed directly,... # but may be overridden in exception subclasses... x, y = inst.args # unpack args... print('x =', x) ... print('y =', y) ...<class 'Exception'>('spam', 'eggs') ('spam', 'eggs') x = spam y = eggs🎜🎜
try
문의 실행 순서는 다음과 같습니다. 🎜try
를 실행합니다. 그리고 try
문이 실행됩니다🎜out
뒤의 예외와 일치할 수 있는 경우, Except 절이 실행되고 try
문 뒤의 문이 계속 실행됩니다. 🎜try
문에 예외가 발생합니다. 처리되지 않은 예외 🎜로 인해 프로그램이 종료되고 위의 예와 같은 메시지가 인쇄됩니다. 🎜try
문에는 다양한 예외에 대한 처리 방법을 지정하기 위해 여러 개의 Except 절이 있을 수 있습니다. 최대 하나의 Except 절이 실행됩니다. 핸들러는 해당 try 절에서 발생하는 예외만 처리하며 동일한 try
문의 다른 핸들러에서 발생하는 예외는 처리하지 않습니다. Except 절은 다음과 같이 괄호로 묶인 튜플을 사용하여 여러 예외를 나열할 수 있습니다. 🎜🎜>>> def this_fails(): ... x = 1/0...>>> try: ... this_fails() ... except ZeroDivisionError as err: ... print('Handling run-time error:', err) ... Handling run-time error: division by zero🎜🎜
Except
절의 클래스는 해당 하위 클래스 또는 자체 유형 예외인 유형과 일치할 수 있습니다(그러나 반대의 경우에는 제외 절의 하위 클래스가 상위 클래스 예외와 일치하지 않습니다. 예를 들어, 다음 코드는 B, C, D를 순서대로 인쇄합니다. 🎜🎜>>> raise NameError('HiThere') Traceback (most recent call last): File "<stdin>", line 1, in <module>NameError: HiThere🎜🎜 위의 제외 절이 역순으로 작성된 경우(
out B
먼저) B, B, B가 인쇄됩니다 - 첫 번째 제외 절의 일치가 트리거됩니다. 🎜🎜마지막 제외 절에서는 예외 이름을 와일드카드로 생략할 수 있습니다. 그렇게 하면 다른 실제 프로그램 오류가 숨겨질 수 있으므로 주의해서 사용하십시오. 오류 메시지를 인쇄하고 다시 발생시키는 데에도 사용할 수 있습니다(호출자가 예외를 처리할 수 있도록 허용). 🎜🎜raise ValueError # shorthand for 'raise ValueError()'🎜🎜
try
...out
문에는 선택 사항인 🎜 else가 있습니다. 절🎜, 이 절은 모든 절 뒤에만 나타날 수 있습니다. try 절이 예외를 발생시키지 않을 때 코드 섹션을 실행해야 하는 경우 else 절을 사용하는 것이 유용합니다(주석: else
절은 return, break, 계속
) . 예: 🎜🎜>>> try: ... raise NameError('HiThere') ... except NameError: ... print('An exception flew by!') ... raise... An exception flew by!Traceback (most recent call last): File "<stdin>", line 2, in <module>NameError: HiThere🎜
使用 else
子句比在 try
子句中附加代码要好,因为这样可以避免 try
... except
意外的捕获的本不属于它们保护的那些代码抛出的异常。
异常发生时,可以携带与之关联的值,也称作异常参数。异常参数是否可以存在以及其类型取决于异常类型。
except子句可以在异常名字后指定变量。该变量绑定到异常实例,异常参数存储在instance.args
属性中。方便起见,异常实例定义了__str__()
以便异常参数可以直接打印,而不是使用.args
引用。也可以首先实例化异常,并在抛出它之前加入任何想要的参数:
>>> try: ... raise Exception('spam', 'eggs') ... except Exception as inst: ... print(type(inst)) # the exception instance... print(inst.args) # arguments stored in .args... print(inst) # __str__ allows args to be printed directly,... # but may be overridden in exception subclasses... x, y = inst.args # unpack args... print('x =', x) ... print('y =', y) ...<class 'Exception'>('spam', 'eggs') ('spam', 'eggs') x = spam y = eggs
对于未处理异常来说,如果它有参数,那么参数将在错误信息的后面部分(详细信息部分)打印。
异常处理程序不仅仅会处理即时发生在try子句中的异常,还会处理try子句中直接或者间接调用的函数中发生的异常。例如:
>>> def this_fails(): ... x = 1/0...>>> try: ... this_fails() ... except ZeroDivisionError as err: ... print('Handling run-time error:', err) ... Handling run-time error: division by zero
raise语句允许程序员强制发生异常。例如:
>>> raise NameError('HiThere') Traceback (most recent call last): File "<stdin>", line 1, in <module>NameError: HiThere
raise
的唯一参数指定要抛出的异常。参数必须是异常实例或者异常类(继承自Exception类的子类)。如果参数是异常类型,会隐式使用无参方式调用异常的构造器初始化一个异常实例:
raise ValueError # shorthand for 'raise ValueError()'
如果需要捕获异常但是不处理,一种更简单形式的raise
语句允许重新抛出这个异常:
>>> try: ... raise NameError('HiThere') ... except NameError: ... print('An exception flew by!') ... raise... An exception flew by!Traceback (most recent call last): File "<stdin>", line 2, in <module>NameError: HiThere
程序中可以通过创建新异常类的方式提出自己的异常(参见Classes
获取Python类的更多信息)。异常必须直接或者间接继承自Exception
类。
自定义异常类拥有其他类的功能,但通常需要保持其简洁性,只提供几个供异常处理程序提取错误信息的属性。需要创建一个抛出若干不同错误的模块时,比较好的实践是,为定义在这个模块中的异常创建父类,由子类创建对应不同错误的具体异常:
class Error(Exception):"""Base class for exceptions in this module."""passclass InputError(Error):"""Exception raised for errors in the input. Attributes: expression -- input expression in which the error occurred message -- explanation of the error """def __init__(self, expression, message):self.expression = expressionself.message = messageclass TransitionError(Error):"""Raised when an operation attempts a state transition that's not allowed. Attributes: previous -- state at beginning of transition next -- attempted new state message -- explanation of why the specific transition is not allowed """def __init__(self, previous, next, message):self.previous = previousself.next = nextself.message = message
与标准异常类类似,大多数异常的名字都以"Error"结尾。
许多标准模块都定义了自己的异常,这些异常对应模块中定义的函数中可能发生的错误。更多信息参考Classes。
try
语句有可选的在任何情况下都会执行的子句,可用于定义清理动作。例如:
>>> try: ... raise KeyboardInterrupt... finally: ... print('Goodbye, world!') ... Goodbye, world!KeyboardInterruptTraceback (most recent call last): File "<stdin>", line 2, in <module>
无论是否有异常发生,finally子句在离开try
语句之前总是会执行。当try
子句中有异常发生并且没有被except
子句处理(或者异常发生在except
子句或else
子句),finally
子句执行之后,这些异常会重新抛出。当try
语句的其他子句通过break, continue
或者return
语句离开时,finally
子句也会执行。以下是较为复杂的示例:
>>> def divide(x, y): ... try: ... result = x / y ... except ZeroDivisionError: ... print("division by zero!") ... else: ... print("result is", result) ... finally: ... print("executing finally clause") ...>>> divide(2, 1) result is 2.0executing finally clause>>> divide(2, 0) division by zero!executing finally clause>>> divide("2", "1") executing finally clause Traceback (most recent call last): File "<stdin>", line 1, in <module> File "<stdin>", line 3, in divideTypeError: unsupported operand type(s) for /: 'str' and 'str'
正如所见的那样,finally
子句在任何情况下都会执行。两个字符串相除产生的TypeError
异常没有被except
子句处理,因此在finally
子句执行完毕之后被重新抛出。
在实践应用中,finally
子句用来释放外部资源(比如文件和网络连接),不论资源的使用是否成功。
一些对象定义了标准的清理动作,当不再需要这些对象时,会执行清理动作,而不论使用对象的操作是否成功。以下示例读取文件并打印内容到显示器:
for line in open("myfile.txt"):print(line, end="")
这段代码的问题是:当代码执行完毕后,文件会保留打开状态一段不确定的时间。这在简单的脚本中不是什么大问题,但是在大型的应用程序中会出问题。with
语句使得像文件一样的对象可以被立即准确回收。
with open("myfile.txt") as f:for line in f:print(line, end="")
语句执行后,即使在执行代码时遇到问题,文件f总是会被关闭。像文件一样提供预定义清理动作的对象会在其说明文档中指示这点。
위 내용은 Python의 두 가지 오류 유형 소개의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!