Python:Python3 异常和文件
Python3 异常和文件操作全解析 📄🔥
本文将系统讲解 Python3 中的异常处理机制与文件操作方法,配合代码实例,让你轻松掌握这一关键基础知识点。无论是编写健壮程序,还是进行数据处理,都是绕不开的重要环节!
什么是异常
程序运行期间发生的错误称为异常(Exception)。当Python脚本遇到无法正常处理的情况时,会抛出异常对象。
常见异常类型
异常类型 | 描述 |
---|---|
FileNotFoundError | 文件未找到错误 |
TypeError | 类型不匹配错误 |
ValueError | 值错误 |
ZeroDivisionError | 除以零错误 |
# 触发ValueError的示例
try:int('abc')
except ValueError as e:print(f"捕获到异常:{e}")
异常处理的方法
Python 使用 try...except
结构来处理异常,以避免程序崩溃。
基本语法:
try:# 有可能抛出异常的代码result = 10 / 0
except ZeroDivisionError:print("除数不能为零!")
捕获多个异常:
try:num = int(input("请输入一个整数:"))result = 10 / num
except ZeroDivisionError:print("除数不能为零")
except ValueError:print("输入的不是有效整数")
使用 as
关键字获取异常信息:
try:f = open("nofile.txt")
except FileNotFoundError as e:print("错误信息:", e)
异常中的 finally 语句
在 Python 的异常处理机制中,finally
语句是一个非常重要的组成部分。finally
语句的作用是无论try
块中的代码是否发生异常,也无论异常是否被except
块捕获,finally
块中的代码都会被执行。这使得finally
语句特别适用于需要执行清理操作的场景,比如关闭文件、释放锁或网络连接等。
try - finally
语句的基本语法结构如下:
try:# 可能会引发异常的代码pass
finally:# 无论是否发生异常,都会执行的代码pass
try - finally
语句结构的执行过程如下:
1. 执行 try 块代码:首先执行try
块内的逻辑。
2. 检查异常:
- 无异常:若
try
块执行后无异常,则跳过except
块(如果有except
块的话),直接执行finally
。 - 有异常:若
try
块执行中发生异常,Python 会立刻暂停当前流程(try
块中剩下未运行的代码),转而去执行finally
块。
3.执行 finally 块:finally
块的代码必定运行,无论try
块代码是否发生异常,或是否有return
、break
、continue
等等控制流语句。
示例 1:文件操作中的资源清理
在文件操作中,我们打开文件后,无论读取或写入过程是否发生异常,都需要关闭文件以释放资源。这时finally
语句就非常有用。
fd = open("test.txt", 'r')
try:content = fd.read()print(content)
finally:fd.close()
在上述代码中,try
块尝试读取文件内容。如果在读取过程中发生异常,比如文件损坏无法读取,程序会立即跳转到finally
块中关闭文件。如果读取过程没有异常,同样会执行finally
块中的关闭文件操作,确保文件资源被正确释放。
示例 2:覆盖返回值或覆盖异常
如果finally
代码块中有return
或抛出新的异常,则会覆盖try
或except
块中的结果。因此,在finally
块中应尽量避免引发异常,除非有充分的理由这么做。
def sub_func():try:return "from try code-blocks"finally:return "from finally code-blocks"# 调用函数sub_func()
info = sub_func()
print(info)
运行上述代码,输出结果为from finally code-blocks
。这是因为finally
块中的return
语句覆盖了try
块中的return
语句。
示例 3:异常未捕获时的情况
如果在try
块中发生了异常并且该异常未被捕获(即没有except
异常处理块的代码),则运行了finally
块的代码后才抛出异常。
try:value1 = 20 / 0 # 发生异常后,try代码块后续代码不会执行value2 = 10 * 1print("try code-blocks executed.")
finally:print("finally code-blocks executed.")
运行上述代码,首先会执行try
块中的20 / 0
引发ZeroDivisionError
异常,此时try
块中后续的value2 = 10 * 1
和print("try code-blocks executed.")
不会执行,而是直接跳转到finally
块中执行print("finally code-blocks executed.")
,然后再抛出ZeroDivisionError
异常。
异常是怎么引发的
异常的引发通常是由于程序在运行过程中遇到了无法正常处理的情况,主要可以分为代码逻辑错误和外部因素这两大原因。下面我们详细分析异常是如何引发的,并结合具体例子进行说明。
代码逻辑错误
1. 条件判断错误:在编写程序时,如果条件判断的逻辑不正确,可能会导致程序执行到不期望的分支,从而引发异常。例如:
num = 10
if num > 20:result = num / 0 # 这里会引发ZeroDivisionError异常,因为逻辑错误导致进入了不该执行的分支
else:result = num * 2
上述代码中,由于条件判断num > 20
为False
,正常情况下应该执行else
分支。但如果在开发过程中,错误地将条件判断写错,导致进入了if
分支,就会执行num / 0
,从而引发ZeroDivisionError
异常。
2. 循环控制问题:循环控制变量的更新或循环结束条件设置不当,可能会导致无限循环或者访问越界等问题,进而引发异常。例如:
my_list = [1, 2, 3]
index = 0
while index < len(my_list):print(my_list[index])index -= 1 # 这里应该是index += 1,由于逻辑错误导致无限循环访问越界
在这个例子中,原本应该是index += 1
来遍历列表,但写成了index -= 1
,这会导致index
永远不会大于或等于列表的长度,从而进入无限循环,并且随着循环的进行,index
会变成负数,导致访问列表时出现IndexError
异常。
3. 函数调用错误:在调用函数时,如果传递的参数数量、类型或顺序不正确,可能会引发异常。例如:
def add_numbers(a, b):return a + bresult = add_numbers(1, '2') # 这里会引发TypeError异常,因为参数类型不匹配
上述代码中,add_numbers
函数期望接收两个数字类型的参数,但实际传递了一个整数和一个字符串,这会导致在执行加法操作时引发TypeError
异常,因为整数和字符串不能直接相加。
外部因素
1. 文件操作相关:当进行文件操作时,如打开、读取、写入文件等,如果文件不存在、权限不足或文件被其他程序占用,都可能引发异常。例如:
try:with open('nonexistent_file.txt',