Python 写代码时,错误是常有的事。为了让程序更健壮,我们需要学会如何处理这些错误。这篇文章会带你从零开始了解 Python 的错误处理。
参考文章:
- Python 错误处理基础 | 简单一点学习 easyeasy.me
- Python 错误处理高级编程 | 简单一点学习 easyeasy.me
- 什么是异常?
- 常见的 Python 异常类型
- 基本的 try-except 结构
- 捕获特定异常
- 使用 else 和 finally
- 抛出异常
- 自定义异常
- 错误处理的最佳实践
1. 什么是异常?
异常(Exception)就是程序运行时发生的错误,比如试图打开一个不存在的文件、除以零或者输入了非法数据。Python 会抛出异常来告诉你“嘿,这里出问题了!”。如果不处理,程序会直接崩溃,打印一堆错误信息(就是 traceback)。
异常处理的核心是用代码“捕获”这些错误,防止程序崩掉,还能给用户友好的提示。
2. 常见的 Python 异常类型
Python 内置了很多异常类型,了解它们能帮你更好地处理问题。以下是一些常见的异常:
- ZeroDivisionError:除以零时触发。
- FileNotFoundError:试图打开不存在的文件。
- ValueError:函数接收到不合适的参数,比如把字符串传给 int()。
- TypeError:类型错误,比如试图把字符串和整数相加。
- IndexError:列表索引超出范围。
- KeyError:访问字典中不存在的键。
这些异常都是 Python 的内置类,继承自 Exception 类。你可以在代码中捕获它们。
3. 基本的 try-except 结构
Python 用 try-except 块来处理异常。基本思路是:把可能出错的代码放进 try 块,如果出错了,就跳到 except 块执行。
来看个简单的例子:- try:
- num = int(input("请输入一个数字: "))
- result = 10 / num
- print(f"10 除以 {num} 等于 {result}")
- except:
- print("出错了!可能输入了非法数字或除以零。")
复制代码 运行这段代码:
- 如果用户输入一个合法数字,比如 5,程序会打印 10 除以 5 等于 2.0。
- 如果输入了非数字(比如 "abc")或 0,程序会跳到 except 块,打印错误提示,而不会崩溃。
这个 except 是个“通吃”模式,捕获所有异常。但最好别这样用,因为你不知道具体出了啥问题。接下来我们改进它。
4. 捕获特定异常
与其捕获所有异常,不如只捕获你关心的特定异常类型。这样能更精准地处理问题。Python 允许在 except 后指定异常类型。
改写上面的例子:- try:
- num = int(input("请输入一个数字: "))
- result = 10 / num
- print(f"10 除以 {num} 等于 {result}")
- except ValueError:
- print("请输入一个合法的数字!")
- except ZeroDivisionError:
- print("不能除以零!")
复制代码 现在:
- 如果输入 "abc",会触发 ValueError,打印“请输入一个合法的数字!”。
- 如果输入 0,会触发 ZeroDivisionError,打印“不能除以零!”。
- 其他异常(比如 KeyboardInterrupt)不会被捕获,程序会照常报错。
你可以捕获多个异常,也可以把异常对象保存下来,查看具体错误信息:- try:
- num = int(input("请输入一个数字: "))
- result = 10 / num
- except ValueError as e:
- print(f"输入错误: {e}")
- except ZeroDivisionError as e:
- print(f"除零错误: {e}")
复制代码 这里 as e 把异常对象存到变量 e,可以打印更详细的错误信息。
5. 使用 else 和 finally
try-except 还有两个好用的伙伴:else 和 finally。
- else:如果 try 块没出错,就执行 else 块。适合放那些依赖 try 块成功的代码。
- finally:不管 try 块有没有出错,finally 块都会执行。常用来做清理工作,比如关闭文件。
来看个例子:- try:
- num = int(input("请输入一个数字: "))
- result = 10 / num
- except ValueError:
- print("请输入合法数字!")
- except ZeroDivisionError:
- print("不能除以零!")
- else:
- print(f"计算成功!结果是: {result}")
- finally:
- print("程序结束,清理完成!")
复制代码 运行效果:
- 输入 5:打印“计算成功!结果是: 2.0”和“程序结束,清理完成!”。
- 输入 "abc":打印“请输入合法数字!”和“程序结束,清理完成!”。
- 输入 0:打印“不能除以零!”和“程序结束,清理完成!”。
finally 特别适合确保资源被正确释放,比如:- try:
- file = open("data.txt", "r")
- content = file.read()
- except FileNotFoundError:
- print("文件没找到!")
- else:
- print(f"文件内容: {content}")
- finally:
- try:
- file.close()
- print("文件已关闭")
- except NameError:
- print("文件从未打开,无需关闭")
复制代码 这里 finally 确保文件被关闭,即使文件没找到(导致 file 未定义)也能安全处理。
6. 抛出异常
有时候你想主动抛出异常,比如检查到用户输入不符合要求。可以用 raise 关键字抛出异常。
例子:- def divide(a, b):
- if b == 0:
- raise ZeroDivisionError("除数不能为零!")
- return a / b
- try:
- result = divide(10, 0)
- except ZeroDivisionError as e:
- print(f"错误: {e}")
复制代码 运行会打印“错误: 除数不能为零!”。你可以用 raise 抛出任何内置异常,或者直接用 raise Exception("自定义消息")。
还可以“重新抛出”异常:- try:
- num = int(input("请输入一个数字: "))
- except ValueError as e:
- print(f"输入错误: {e}")
- raise # 重新抛出原异常
复制代码 这样会在处理完错误后,继续把异常抛给上层调用者。
7. 自定义异常
当内置异常不够用时,你可以定义自己的异常类。自定义异常通常继承自 Exception 类。
例子:- class CustomValueError(Exception):
- """自定义异常,用于值不符合要求"""
- pass
- def check_age(age):
- if age < 0 or age > 150:
- raise CustomValueError("年龄必须在0到150之间!")
- return age
- try:
- age = check_age(200)
- except CustomValueError as e:
- print(f"错误: {e}")
复制代码 运行会打印“错误: 年龄必须在0到150之间!”。自定义异常让代码更清晰,尤其在复杂项目中。
8. 错误处理的最佳实践
最后,分享一些写错误处理代码的建议:
- 捕获特定异常:别用“通吃”的 except,尽量指定具体的异常类型。
- 提供清晰的错误信息:用户看到错误时,应该知道问题出在哪,怎么解决。
- 用 else 和 finally 合理组织代码:else 放依赖 try 成功的逻辑,finally 放清理工作。
- 别滥用异常:异常是用来处理“异常情况”的,别用它来控制正常流程(比如代替 if 判断)。
- 记录错误日志:在实际项目中,用 logging 模块记录异常,方便调试。
例子(带日志):- import logging
- logging.basicConfig(level=logging.ERROR, filename="app.log")
- try:
- num = int(input("请输入一个数字: "))
- result = 10 / num
- except ValueError as e:
- logging.error(f"输入错误: {e}")
- print("请输入合法数字!")
- except ZeroDivisionError as e:
- logging.error(f"除零错误: {e}")
- print("不能除以零!")
- else:
- print(f"结果: {result}")
- finally:
- print("程序结束!")
复制代码 这个例子会把错误记录到 app.log 文件,方便排查问题。
来源:程序园用户自行投稿发布,如果侵权,请联系站长删除
免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作! |