Python & AI Tutorials Logo
Python 编程

24. 理解错误与回溯

错误是编程中不可避免的一部分。每一位程序员,从初学者到专家,都会经常遇到它们。面对错误时是挣扎还是从中学习,差别在于你是否理解:当出问题时,Python 正在试图告诉你什么。

当 Python 在你的代码中遇到问题时,它不会悄无声息地停下来——它会提供详细信息,说明哪里出了问题、发生在什么位置,并且通常会暗示为什么会这样。学会阅读并解读这些错误信息,是你作为程序员能培养的最有价值的技能之一。

在本章中,我们将探索你会遇到的两大类错误:语法错误(syntax errors)(你写代码方式的问题)和 运行时异常(runtime exceptions)(代码运行时出现的问题)。我们将学习阅读 回溯(tracebacks)——Python 详细的错误报告——并理解异常如何改变程序的正常流程。最重要的是,我们将培养一种调试思维,把错误视为不是失败,而是帮助你写出更好代码的宝贵信息。

24.1) 语法错误 vs 运行时异常

Python 会区分你代码中两种本质上不同的问题:语法错误和运行时异常。理解这种区别能帮助你更快诊断问题,并知道应该去哪里找解决方案。

24.1.1) 什么是语法错误

当 Python 无法理解你的代码,因为它违反了这门语言的语法规则时,就会发生 语法错误(syntax error)。就像 “The cat sat on the” 是一个不完整的英文句子一样,带有语法错误的代码是不完整或格式错误的 Python,解释器无法解析。

语法错误会在你的程序运行之前被检测到。Python 会先通读你的整个脚本,检查它是否遵循语言规则。如果发现语法错误,它会拒绝执行任何代码——即使是那些正确的部分也不执行。

下面是一个简单示例:

python
# 警告:语法错误——仅用于演示
# 错误:if 语句后缺少冒号
age = 25
if age >= 18
    print("You are an adult")

当你尝试运行这段代码时,Python 会立刻报告:

  File "example.py", line 3
    if age >= 18
                ^
SyntaxError: expected ':'

请注意这条错误信息的几个关键特征:

  1. 文件和行号:Python 准确告诉你它在哪里发现了问题(line 3
  2. 可视化指示:插入符号(^)指向 Python 开始困惑的位置
  3. 错误类型SyntaxError 清楚表明这是一个语法问题
  4. 有用提示expected ':' 告诉你缺少了什么

代码永远不会运行,因为 Python 甚至无法开始执行——语法是无效的。

我们再看另一个常见的语法错误:

python
# 警告:语法错误——仅用于演示
# 错误:括号不匹配
numbers = [1, 2, 3, 4, 5]
total = sum(numbers
print(f"Total: {total}")

Python 报告:

  File "example.py", line 2
    total = sum(numbers
               ^
SyntaxError: '(' was never closed

这里,Python 检测到我们在第 2 行打开了一个括号,但从未关闭它。错误在第 2 行报告(未关闭括号所在位置),插入符号指向 Python 期望找到右括号的位置。

语法错误的关键特征:

  • 在任何代码运行之前被检测到
  • 阻止整个程序执行
  • 通常表示拼写错误、缺少标点或缩进不正确
  • 报错位置可能会稍微晚于真正的错误位置

24.1.2) 什么是运行时异常

运行时异常(runtime exception)(或简称 “异常(exception)”)是在语法正确的代码在执行过程中遇到问题时发生的。代码在语法上是正确的 Python,但程序实际运行时出了问题。

与语法错误不同,异常发生在你的程序运行过程中。Python 已经成功解析了你的代码并开始执行,但随后遇到了它无法处理的情况。

下面是一个简单示例:

python
# 这段代码语法有效,但会抛出异常
numbers = [10, 20, 30]
print(numbers[0])  # Output: 10
print(numbers[5])  # 这一行会引发 IndexError
print("This line never executes")

Output:

10
Traceback (most recent call last):
  File "example.py", line 3, in <module>
    print(numbers[5])
          ~~~~~~~^^^
IndexError: list index out of range

注意发生了什么:

  1. 第一条 print 语句成功执行(我们看到了 10
  2. 第二条 print 试图访问索引 5,但它不存在
  3. Python 抛出了一个 IndexError 异常
  4. 程序停止,第三条 print 永远不会执行

代码在语法上是正确的——Python 理解我们想做什么完全没有问题。问题是在执行过程中,当我们试图访问一个不存在的列表(list)元素时出现的。

下面是另一个示例,展示不同类型的运行时异常:

python
# 语法有效,但运行时会发生除以零
def calculate_average(total, count):
    return total / count
 
# 这些都能正常工作
print(calculate_average(100, 4))  # Output: 25.0
print(calculate_average(75, 3))   # Output: 25.0
 
# 这会引发异常
print(calculate_average(50, 0))   # ZeroDivisionError

Output:

25.0
25.0
Traceback (most recent call last):
  File "example.py", line 8, in <module>
    print(calculate_average(50, 0))
          ^^^^^^^^^^^^^^^^^^^^^^^^
  File "example.py", line 2, in calculate_average
    return total / count
           ~~~~~~^~~~~~~
ZeroDivisionError: division by zero

这个函数前两次都工作得很好,但第三次调用时,我们把 0 作为 count 传入,导致除以零。Python 直到代码用这些特定值真正运行时才发现这个问题。

运行时异常的关键特征:

  • 发生在程序执行期间
  • 代码语法有效
  • 通常依赖于特定数据或条件
  • 程序会运行到异常发生的位置
  • 不同输入可能导致不同异常(或者根本不出错)

24.1.3) 对比语法错误与运行时异常

让我们把两类错误放在一起对比,以理解它们的差异:

python
# 示例 1:语法错误
# 错误:缺少结束引号
print("Program started!")
message = "Hello, world
print(message)

这会立刻产生语法错误:

  File "example.py", line 4
    message = "Hello, world
              ^
SyntaxError: unterminated string literal (detected at line 4)

重要:请注意你不会在输出中看到 “Program started!”。Python 在运行任何代码之前就检测到了语法错误。

现在对比一个运行时异常:

python
# 示例 2:运行时异常
# 语法有效,但变量不存在
print("Program started!")
message = "Hello, world"
print(mesage)  # 拼写错误:'mesage' 而不是 'message'

Output:

Program started!
Traceback (most recent call last):
  File "example.py", line 5, in <module>
    print(mesage)
          ^^^^^^
NameError: name 'mesage' is not defined

重要:这一次你确实会在输出中看到 “Program started!”。Python 成功运行了前两条 print 以及赋值语句(第 3-4 行),但在第 5 行试图找到 mesage 时遇到了问题。

关键差异:在第一个示例中,Python 甚至从未尝试运行代码——它在解析阶段发现了语法错误。在第二个示例中,Python 成功开始执行程序,并在遇到运行时错误之前运行了好几行。

Python 读取你的代码

语法有效吗?

语法错误
程序永远不会运行

程序开始执行

执行过程中
出现问题吗?

程序成功
完成

运行时异常
程序停止

24.2) 常见内置异常类型

Python 有许多内置异常类型,每一种都代表某类特定问题。学会识别这些常见异常能帮助你快速理解哪里出了问题以及如何修复。每种异常类型都有一个描述性的名称,会暗示问题所在。

24.2.1) NameError:使用未定义的名称

当你尝试使用一个 Python 不认识的变量、函数(function)或其他名称时,就会发生 NameError。这通常意味着你忘了定义某个东西、拼错了名称,或者在它被创建之前就试图使用它。

python
# 示例 1:忘记定义变量
print(greeting)  # NameError: name 'greeting' is not defined

Output:

Traceback (most recent call last):
  File "example.py", line 2, in <module>
    print(greeting)
          ^^^^^^^^
NameError: name 'greeting' is not defined

Python 在告诉你它不知道 greeting 是什么。你需要先创建它:

python
# 正确版本
greeting = "Hello, Python!"
print(greeting)  # Output: Hello, Python!

下面是一个更隐蔽的示例:拼写错误:

python
# 示例 2:变量名拼写错误
user_name = "Alice"
age = 30
 
print(f"{username} is {age} years old")  # NameError: name 'username' is not defined

我们定义了 user_name(带下划线),但试图使用 username(不带下划线)。Python 会把它们视为完全不同的名称。

24.2.2) TypeError:对操作使用了错误类型

当你尝试对错误类型的值执行某个操作时,会发生 TypeError。例如,你不能把字符串和整数相加,或者调用一个并非函数的对象。

python
# 示例 1:混用不兼容的类型
age = 25
message = "You are " + age + " years old"  # TypeError

Output:

Traceback (most recent call last):
  File "example.py", line 2, in <module>
    message = "You are " + age + " years old"
              ~~~~~~~~~~~~^~~~~
TypeError: can only concatenate str (not "int") to str

Python 在告诉你 + 运算符可以把字符串拼接到字符串,但不能把字符串拼接到整数。你需要把整数转换为字符串:

python
# 正确版本
age = 25
message = "You are " + str(age) + " years old"
print(message)  # Output: You are 25 years old

当你向函数传递错误数量的参数时,也会发生 TypeError:

python
# 示例 3:参数数量错误
def calculate_area(length, width):
    return length * width
 
area = calculate_area(5)  # TypeError: missing 1 required positional argument

Output:

Traceback (most recent call last):
  File "example.py", line 4, in <module>
    area = calculate_area(5)
TypeError: calculate_area() missing 1 required positional argument: 'width'

该函数需要两个参数,但我们只提供了一个。

24.2.3) ValueError:类型正确,值不合适

当你传入了类型正确的值,但这个值本身不适合该操作时,就会发生 ValueError。类型是对的,但具体的值在该上下文中不合理。

python
# 示例 1:将无效字符串转换为整数
user_input = "twenty-five"
age = int(user_input)  # ValueError: invalid literal for int()

Output:

Traceback (most recent call last):
  File "example.py", line 2, in <module>
    age = int(user_input)
ValueError: invalid literal for int() with base 10: 'twenty-five'

int() 函数需要一个字符串,我们也确实给了它一个字符串——因此类型是正确的。但字符串 "twenty-five" 无法转换为整数,因为它包含字母。字符串 "25" 就没问题:

python
# 正确版本
user_input = "25"
age = int(user_input)
print(age)  # Output: 25

ValueError 也会出现在列表方法中:

python
# 示例 3:移除不存在的元素
fruits = ["apple", "banana", "orange"]
fruits.remove("grape")  # ValueError: 'grape' is not in list

Output:

Traceback (most recent call last):
  File "example.py", line 2, in <module>
    fruits.remove("grape")
    ~~~~~~~~~~~~~^^^^^^^^^
ValueError: list.remove(x): x not in list

remove() 方法期望传入的值存在于列表中。我们应该先检查:

python
# 正确版本
fruits = ["apple", "banana", "orange"]
if "grape" in fruits:
    fruits.remove("grape")
else:
    print("Grape not found in list")  # Output: Grape not found in list

24.2.4) IndexError:无效的序列索引

当你试图使用一个不存在的索引来访问序列(列表、元组、字符串)时,就会发生 IndexError。请记住 Python 使用从 0 开始的索引,有效索引范围是 0len(sequence) - 1

python
# 示例 1:索引过大
colors = ["red", "green", "blue"]
print(colors[0])  # Output: red
print(colors[3])  # IndexError: list index out of range

Output:

red
Traceback (most recent call last):
  File "example.py", line 3, in <module>
    print(colors[3])
          ~~~~~~^^^
IndexError: list index out of range

这个列表有三个元素,索引分别是 0、1、2。索引 3 并不存在。这是一个非常常见的错误:忘记索引从 0 开始。

python
# 正确版本
colors = ["red", "green", "blue"]
print(colors[2])  # Output: blue (the third element)

24.2.5) KeyError:缺少字典键

当你试图访问一个不存在的字典键(key)时,会发生 KeyError。与可以检查长度的列表不同,字典可以有任意键,因此在访问之前你需要验证键是否存在。

python
# 示例 1:访问不存在的键
student = {
    "name": "Alice",
    "age": 20,
    "major": "Computer Science"
}
 
print(student["name"])   # Output: Alice
print(student["grade"])  # KeyError: 'grade'

Output:

Alice
Traceback (most recent call last):
  File "example.py", line 7, in <module>
    print(student["grade"])
          ~~~~~~~^^^^^^^^^
KeyError: 'grade'

这个字典没有 "grade" 这个键。你可以先检查键是否存在:

python
# 使用 'in' 的正确版本
student = {
    "name": "Alice",
    "age": 20,
    "major": "Computer Science"
}
 
if "grade" in student:
    print(student["grade"])
else:
    print("Grade not available")  # Output: Grade not available

或者使用 get() 方法,它会返回 None(或默认值),而不是抛出错误:

python
# 使用 get() 的替代方案
grade = student.get("grade")
if grade is not None:
    print(f"Grade: {grade}")
else:
    print("Grade not available")  # Output: Grade not available

当处理结构不一致的数据时,KeyError 很常见:

python
# 示例 2:处理多条记录
students = [
    {"name": "Alice", "age": 20, "grade": "A"},
    {"name": "Bob", "age": 21},  # 缺少 'grade' 键
    {"name": "Carol", "age": 19, "grade": "B"}
]
 
for student in students:
    print(f"{student['name']}: {student['grade']}")  # 在 Bob 这里触发 KeyError

Output:

Alice: A
Traceback (most recent call last):
  File "example.py", line 7, in <module>
    print(f"{student['name']}: {student['grade']}")
                                ~~~~~~~^^^^^^^^^
KeyError: 'grade'

使用带默认值的 get() 来优雅处理缺失键:

python
# 正确版本
students = [
    {"name": "Alice", "age": 20, "grade": "A"},
    {"name": "Bob", "age": 21},
    {"name": "Carol", "age": 19, "grade": "B"}
]
 
for student in students:
    grade = student.get("grade", "Not assigned")
    print(f"{student['name']}: {grade}")

Output:

Alice: A
Bob: Not assigned
Carol: B

24.2.6) AttributeError:无效的属性访问

当你试图访问一个对象上不存在的属性或方法时,会发生 AttributeError。这通常发生在你混淆了不同类型的方法,或拼错了属性名。

python
# 示例 1:对类型使用了错误的方法
numbers = [1, 2, 3, 4, 5]
numbers.append(6)  # 这能工作——列表有 append()
print(numbers)     # Output: [1, 2, 3, 4, 5, 6]
 
text = "Hello"
text.append("!")   # AttributeError: 'str' object has no attribute 'append'

Output:

[1, 2, 3, 4, 5, 6]
Traceback (most recent call last):
  File "example.py", line 6, in <module>
    text.append("!")
    ^^^^^^^^^^^
AttributeError: 'str' object has no attribute 'append'

字符串没有 append() 方法,因为字符串是不可变的。你需要使用拼接或其他字符串方法:

python
# 正确版本
text = "Hello"
text = text + "!"  # 拼接
print(text)        # Output: Hello!

AttributeError 也会因拼写错误而发生:

python
# 示例 2:方法名拼写错误
message = "Python Programming"
result = message.uppper()  # AttributeError: 'str' object has no attribute 'uppper'

Output:

Traceback (most recent call last):
  File "example.py", line 2, in <module>
    result = message.uppper()
             ^^^^^^^^^^^^^^
AttributeError: 'str' object has no attribute 'uppper'. Did you mean: 'upper'?

注意 Python 3.10+ 通常会建议正确拼写!正确的方法是 upper()

python
# 正确版本
message = "Python Programming"
result = message.upper()
print(result)  # Output: PYTHON PROGRAMMING

24.2.7) ZeroDivisionError:除以零

当你试图用一个数除以零时,会发生 ZeroDivisionError,因为这在数学上是未定义的。这常见于用户输入或计算值中出现了你没预料到的 0。

python
# 示例 1:直接除以零
result = 10 / 0  # ZeroDivisionError: division by zero

Output:

Traceback (most recent call last):
  File "example.py", line 1, in <module>
    result = 10 / 0
             ~~~^~~
ZeroDivisionError: division by zero

这同样适用于整除和取模:

python
# 示例 2:其他除法运算
a = 10 // 0  # ZeroDivisionError
b = 10 % 0   # ZeroDivisionError

一个更现实的例子涉及计算:

python
# 示例 3:计算平均值
def calculate_average(numbers):
    total = sum(numbers)
    count = len(numbers)
    return total / count
 
scores = [85, 90, 78, 92]
print(calculate_average(scores))  # Output: 86.25
 
empty_scores = []
print(calculate_average(empty_scores))  # ZeroDivisionError

Output:

86.25
Traceback (most recent call last):
  File "example.py", line 9, in <module>
    print(calculate_average(empty_scores))
          ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
  File "example.py", line 4, in calculate_average
    return total / count
           ~~~~~~^~~~~~~
ZeroDivisionError: division by zero

空列表的长度为 0,导致除以零。务必检查这种情况:

python
# 正确版本
def calculate_average(numbers):
    if len(numbers) == 0:
        return 0  # 或返回 None,或抛出更具描述性的错误
    
    total = sum(numbers)
    count = len(numbers)
    return total / count
 
scores = [85, 90, 78, 92]
print(calculate_average(scores))  # Output: 86.25
 
empty_scores = []
print(calculate_average(empty_scores))  # Output: 0

常见异常类型

NameError
未定义的变量/函数

TypeError
操作使用了错误类型

ValueError
类型正确,值不合适

IndexError
无效的序列索引

KeyError
缺少字典键

AttributeError
无效的属性/方法

ZeroDivisionError
除以零

理解这些常见异常类型能帮助你快速诊断问题。当你看到异常时,类型名称会立刻告诉你发生了哪一类问题,而错误信息则提供了关于具体哪里出错的细节。

24.3) 详细阅读与解读回溯信息

当运行时异常发生时,Python 不仅会告诉你哪里错了——它还会提供一个详细的 回溯(traceback),展示你的程序是如何一步步走到那一刻的。学会读回溯信息对有效调试至关重要。回溯就像一条面包屑路径,显示程序在遇到错误之前走过的路线。

24.3.1) 回溯信息的结构

我们从一个简单示例开始,逐一检查回溯的每个部分:

python
# 含错误的简单程序
def calculate_discount(price, discount_percent):
    discount_amount = price * (discount_percent / 100)
    final_price = price - discount_amount
    return final_price
 
def process_order(item_price, discount):
    discounted_price = calculate_discount(item_price, discount)
    tax = discounted_price * 0.08
    total = discounted_price + tax
    return total
 
# 主程序
original_price = "50"  # 哎呀!这里应该是数字
discount_rate = 10
final_cost = process_order(original_price, discount_rate)
print(f"Final cost: ${final_cost:.2f}")

Output:

Traceback (most recent call last):
  File "example.py", line 16, in <module>
    final_cost = process_order(original_price, discount_rate)
                 ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
  File "example.py", line 8, in process_order
    discounted_price = calculate_discount(item_price, discount)
                       ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
  File "example.py", line 2, in calculate_discount
    discount_amount = price * (discount_percent / 100)
                      ~~~~~~^~~~~~~~~~~~~~~~~~~~~~~~~~
TypeError: can't multiply sequence by non-int of type 'float'

让我们拆解这个回溯信息的每个组成部分:

1. 头部:"Traceback (most recent call last):"

这一行告诉你接下来的是回溯信息——一条函数调用记录。"most recent call last" 表示回溯按时间顺序展示:最先调用的函数在最上面,而真正发生错误的位置在最后一行。

2. 调用栈(从上到下阅读):

  File "example.py", line 16, in <module>
    final_cost = process_order(original_price, discount_rate)
                 ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^

这是链条中的第一次函数调用。它显示:

  • 文件名"example.py"——代码所在位置
  • 行号line 16——发起调用的确切行
  • 上下文in <module>——这段代码在顶层(不在任何函数里)
  • 代码:实际执行的那一行
  • 高亮^ 字符指向该行中涉及的具体部分

<module> 上下文表示代码在模块级别运行(脚本的主体部分),而不是在任何函数内部。

  File "example.py", line 8, in process_order
    discounted_price = calculate_discount(item_price, discount)
                       ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^

这是第二次函数调用。process_order 函数从第 16 行被调用,现在我们在该函数的第 8 行,在这里它调用了 calculate_discount

  File "example.py", line 2, in calculate_discount
    discount_amount = price * (discount_percent / 100)
                      ~~~~~~^~~~~~~~~~~~~~~~~~~~~~~~~~

这就是错误实际发生的位置。我们现在在 calculate_discount 函数的第 2 行,这一行导致了问题。

3. 错误信息:

TypeError: can't multiply sequence by non-int of type 'float'

这就是实际发生的错误:

  • 异常类型TypeError——告诉你错误的类别
  • 描述:其余部分具体解释发生了什么

在这个例子中,Python 告诉我们:我们试图把一个序列(这里是字符串)乘以一个 float,这是不允许的。

24.3.2) 从下往上阅读回溯信息

虽然回溯信息是按时间顺序打印的(从上到下),但有经验的程序员往往会从下到上阅读,因为真正的错误在最底部,而上面的行展示了我们是如何走到那里的。

让我们从下到上阅读刚才的回溯:

步骤 1:从错误信息开始

TypeError: can't multiply sequence by non-int of type 'float'

“好的,我们尝试把一个序列乘以 float。这是不允许的。”

步骤 2:看错误发生的位置

  File "example.py", line 2, in calculate_discount
    discount_amount = price * (discount_percent / 100)
                      ~~~~~~^~~~~~~~~~~~~~~~~~~~~~~~~~

“错误发生在 calculate_discount 函数的第 2 行。我们正在把 price 乘以某个东西。”

步骤 3:回溯看看我们是怎么到这里的

  File "example.py", line 8, in process_order
    discounted_price = calculate_discount(item_price, discount)

calculate_discountprocess_order 的第 8 行被调用,并把 item_price 作为 price 参数传入。”

步骤 4:继续向上追溯

  File "example.py", line 16, in <module>
    final_cost = process_order(original_price, discount_rate)

“而 process_order 是在主程序的第 16 行被调用,并把 original_price 作为 item_price 传入。”

步骤 5:找到根本原因

现在我们可以追踪到问题:original_price"50"(一个字符串),它被作为 item_price 传给 process_order,又作为 price 传给 calculate_discount,在这里我们试图把它乘以一个 float。解决方案是让 original_price 变成数字:

python
# 修正版本
def calculate_discount(price, discount_percent):
    discount_amount = price * (discount_percent / 100)
    final_price = price - discount_amount
    return final_price
 
def process_order(item_price, discount):
    discounted_price = calculate_discount(item_price, discount)
    tax = discounted_price * 0.08
    total = discounted_price + tax
    return total
 
# 主程序——修复类型问题
original_price = 50  # 现在它是数字而不是字符串
discount_rate = 10
final_cost = process_order(original_price, discount_rate)
print(f"Final cost: ${final_cost:.2f}")  # Output: Final cost: $48.60

阅读回溯信息

1. 读错误信息
最底一行
2. 找到错误位置
最后一段代码块
3. 追踪调用栈
向上回溯
4. 形成假设
哪里出问题?
5. 验证并修复
测试你的解决方案

理解如何阅读回溯信息,会把它从令人畏惧的一堵文字墙,变成有用的调试工具。每一行都提供了关于程序执行路径的宝贵信息,随着练习,你将能够通过跟随回溯的指引快速定位并修复问题。

24.4) 异常如何改变程序的正常流程

当异常发生时,它不仅仅会停止你的程序——它从根本上改变了程序的执行方式。理解这种行为对编写健壮代码以及理解错误发生时的情况至关重要。

24.4.1) 正常程序流程 vs 异常流程

在正常执行中,Python 会逐行运行你的代码,从上到下:

python
# 正常程序流程
print("Step 1: Starting calculation")
result = 10 + 5
print(f"Step 2: Result is {result}")
final = result * 2
print(f"Step 3: Final value is {final}")
print("Step 4: Program complete")

Output:

Step 1: Starting calculation
Step 2: Result is 15
Step 3: Final value is 30
Step 4: Program complete

每一行都会按顺序执行。现在让我们看看当异常发生时会怎样:

python
# 出现异常时的程序流程
print("Step 1: Starting calculation")
result = 10 / 0  # 这会引发 ZeroDivisionError
print(f"Step 2: Result is {result}")  # 这一行永远不会执行
final = result * 2  # 这一行永远不会执行
print(f"Step 3: Final value is {final}")  # 这一行永远不会执行
print("Step 4: Program complete")  # 这一行永远不会执行

Output:

Step 1: Starting calculation
Traceback (most recent call last):
  File "example.py", line 2, in <module>
    result = 10 / 0
             ~~~^~~
ZeroDivisionError: division by zero

注意只有第一条 print 语句执行了。一旦第 2 行发生异常,Python 就停止执行其余代码。异常打断了正常流程。

24.4.2) 异常沿调用栈向上传播

当异常在某个函数内部发生时,Python 不会只停在该函数——它会沿调用栈向上传播(propagate),直到有东西处理它,或者程序终止。

python
# 示例 1:异常在函数之间传播
def calculate_average(numbers):
    total = sum(numbers)
    count = len(numbers)
    return total / count  # 可能引发 ZeroDivisionError
 
def process_scores(score_list):
    print("Processing scores...")
    avg = calculate_average(score_list)
    print(f"Average calculated: {avg}")
    return avg
 
def main():
    print("Program starting")
    scores = []  # 空列表
    result = process_scores(scores)
    print(f"Final result: {result}")
    print("Program ending")
 
main()

Output:

Program starting
Processing scores...
Traceback (most recent call last):
  File "example.py", line 18, in <module>
    main()
  File "example.py", line 14, in main
    result = process_scores(scores)
             ^^^^^^^^^^^^^^^^^^^^^^^^^^
  File "example.py", line 9, in process_scores
    avg = calculate_average(score_list)
          ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
  File "example.py", line 4, in calculate_average
    return total / count
           ~~~~~~^~~~~~~
ZeroDivisionError: division by zero

让我们追踪发生了什么:

  1. main() 开始执行并打印 “Program starting”
  2. main() 调用 process_scores()
  3. process_scores() 打印 “Processing scores...”
  4. process_scores() 调用 calculate_average()
  5. calculate_average() 尝试除以零
  6. 异常发生并向上传播
    • calculate_average() 立刻停止(没有返回值)
    • 控制权返回 process_scores(),但不是正常返回——异常继续传播
    • process_scores() 立刻停止(calculate_average() 之后的 print 永远不会执行)
    • 控制权返回 main(),但同样异常继续传播
    • main() 立刻停止(process_scores() 之后的 print 永远不会执行)
  7. 程序以回溯信息终止

任何函数中异常之后的代码都不会执行。异常会通过所有函数调用一路“冒泡”,直到到达顶层并终止程序。

24.5) 调试思维:把错误当作信息,而不是失败

编程中最重要的技能之一,不是写出完美无缺的代码——而是学会如何高效地处理不完美的代码。每一位程序员,无论经验水平如何,都会写出产生错误的代码。挣扎的程序员与高效程序员的差别,不在于是否避免错误,而在于他们如何应对错误。

24.5.1) 错误不是失败

当你学习编程时,遇到错误感到沮丧是很自然的。你可能会觉得自己做错了什么,或者觉得自己并没有“学会”。这种心态会适得其反,而且更重要的是,它并不准确。

错误不是失败——它们是反馈。

把错误想象成 GPS 重新规划路线。当你错过一个转弯时,GPS 不会说“你失败了!”它会说“正在重新规划路线”,然后给你新的指引。Python 的错误信息也是一样:它们在告诉你你走的路径行不通,并提供信息帮助你找到可行的路径。

考虑这个简单示例:

python
# 第一次尝试计算平均值
def calculate_average(numbers):
    total = sum(numbers)
    average = total / len(numbers)
    return average
 
scores = []
result = calculate_average(scores)
print(f"Average: {result}")

Output:

Traceback (most recent call last):
  File "example.py", line 8, in <module>
    result = calculate_average(scores)
             ^^^^^^^^^^^^^^^^^^^^^^^^^
  File "example.py", line 4, in calculate_average
    average = total / len(numbers)
              ~~~~~~^~~~~~~~~~~~~~
ZeroDivisionError: division by zero

这个错误并不是在告诉你你是个糟糕的程序员。它告诉你的是具体且有用的信息:“你尝试除以零,这发生在列表为空的时候。你需要处理这种情况。”

掌握了这些信息后,你就可以改进代码:

python
# 基于错误反馈的改进版本
def calculate_average(numbers):
    if len(numbers) == 0:
        return 0  # Or return None, or raise a more descriptive error
    
    total = sum(numbers)
    average = total / len(numbers)
    return average
 
scores = []
result = calculate_average(scores)
print(f"Average: {result}")  # Output: Average: 0

错误帮助你写出了更好的代码。如果没有这个错误,你可能不会意识到你的函数无法处理空列表。

24.5.2) 每一个错误都会教会你一些东西

你遇到的每一个错误,都会教会你一些关于 Python、关于你的代码,或关于编程的一般知识。让我们看几个例子,看看不同错误会教我们什么:

示例 1:学习类型

python
# 尝试相加不兼容的类型
age = 25
message = "You are " + age + " years old"

Output:

TypeError: can only concatenate str (not "int") to str

它教会了什么:Python 有严格的类型规则。你不能在拼接中混用字符串和数字。这个错误让你理解类型兼容性,并引入类型转换的概念。

示例 2:学习数据结构

python
# 尝试像访问列表一样访问字典
student = {"name": "Alice", "age": 20}
first_value = student[0]

Output:

KeyError: 0

它教会了什么:字典使用键,而不是数字索引。这个错误让你理解字典与列表的差异,以及如何正确访问字典的值。

示例 3:学习作用域

python
# 尝试在定义变量之前使用它
def greet():
    print(f"Hello, {name}!")
 
greet()
name = "Alice"

Output:

NameError: name 'name' is not defined

它教会了什么:变量必须在使用之前定义,执行顺序很重要。这个错误让你理解变量作用域以及初始化的重要性。

这些错误中的每一个都提供了具体、可执行的信息,帮助你更好地理解 Python。与其把它们视为障碍,不如把它们当作学习机会。

24.5.3) 拥抱调试思维

职业程序员会花相当一部分时间在调试上。这不是软弱的表现——它是工作核心的一部分。最优秀的程序员不是从不犯错的人;他们是那些会:

  1. 预期错误:他们知道错误会发生,不会感到惊讶或气馁
  2. 认真阅读错误:他们从错误信息中提取最大信息
  3. 系统地调试:他们遵循逻辑流程,而不是随意乱改
  4. 从错误中学习:他们把每次错误都当作更好理解 Python 的机会
  5. 保持好奇:他们问“为什么会发生?”而不只是“怎么修?”

No

Yes

No

Yes

遇到错误

认真阅读错误信息
Carefully

理解哪里
出错了

形成关于原因的
假设

用调试输出验证
假设

假设
正确吗?

实施修复

测试解决方案

是否
正常工作?

从经验中
学习

带着信心
继续前进

记住:每一个错误都是一次学习新东西的机会——关于 Python、关于编程,或关于解决问题。把错误当作宝贵反馈,系统化地处理它们,并为自己的调试成功而感到欣慰。这种心态将贯穿你整个编程旅程。


理解错误与回溯信息,是成为高效 Python 程序员的基础。在本章中,我们学会了区分语法错误(代码结构问题)与运行时异常(执行过程中的问题),识别常见异常类型以及它们所指示的问题,阅读并解读详细回溯信息以找到问题的根本原因,理解异常如何通过沿调用栈向上传播来改变程序流程,并培养一种调试思维,把错误当作宝贵信息而不是失败。

这些技能为下一章打下基础:我们将学习如何使用 tryexcept 块优雅地处理异常,让程序从错误中恢复并继续运行。但在我们能够有效处理异常之前,必须先彻底理解它们——而这正是我们在这里完成的事情。

© 2025. Primesoft Co., Ltd.
support@primesoft.ai