百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术资源 > 正文

Python 异常处理方式汇总,建议收藏!

off999 2024-11-19 08:33 43 浏览 0 评论

本文首发自「慕课网」,想了解更多IT干货内容,程序员圈内热闻,欢迎关注!

作者| 慕课网精英讲师 朱广蔚

程序读文件内容的过程可能会发生错误,例如:要读取的文件不存在。传统的错误处理方式如下:

  • 某个函数 f 在运行过程中可能会发生错误;
  • 函数 f 发生错误时,函数 f 返回错误代码;
  • 在调用函数 f 的地方,需要检查 f 的返回值是否有错。

1. 传统的错误处理方式

1.1 返回错误码

例如,在 C 语言中,函数 open 用于打开一个文件,它的声明如下:

int open(char *path, int mode);
代码块1
  • 参数 path 指定要打开的文件;
  • 参数 mode 指定打开文件的方式:只读、读写;
  • 函数返回一个整数,该整数作为文件的标识符;如果打开文件成功,则返回一个非负的整数;如果打开文件失败,则返回 -1。

因此,通过检查函数 open 的返回值,即可以判断 open 是否成功,示例如下:

int file = open("test.txt", O_RDONLY);
if (file < 0)
    puts("open file failed");
    ...
代码块1234
  • 在第 1 行,函数 open 打开文件 test.txt
  • 在第 2 行,如果函数 open 的返回值小于 0,则表示打开文件失败

1.2 缺点

通过错误代码的方式很容易理解,但是存在一个严重的问题:用户可能忘记了错误检查。例如:

int file = open("test.txt", O_RDONLY);
char buf[1024];
read(file, buf, sizeof(buf));
对 buf 中的数据进行处理;
close(file);
代码块12345
  • 在第 1 行,使用 open 打开文件;在此处忘记对 open 的返回值进行检查;如果文件 test.txt 不存在,则 open 返回 -1,此时 file 为 -1;
  • 在第 3 行,使用 read 读取文件 file,将内容读取到 buf 中;open 的操作失败了,此时 file 为 -1;read 的第一个参数 file 是一个无效的文件标识符;read 的操作必然也是失败的;
  • 在第 4 行,对 buf 中的数据进行处理;open 操作和 open 操作都发生了错误;buf 中的数据是无效数据。

在整个过程中,发生了两次错误:open 文件失败、read 文件失败,但是用户没有得到任何提醒。buf 中的数据是无效的,对读取的数据进行操作是无效的。

2. 异常的处理方式

Python 程序的执行过程中,当发生错误时会引起一个事件,该事件被称为异常。异常会打断程序的正常执行流程,例如,编写程序 control-flow.py:

print('AAA')
100 / 0
print('BBB') # 此行代码不会被执行
代码块123
  • 在第 1 行,打印 AAA;
  • 在第 2 行,100 除以 0;除数是 0,Python 无法执行该条语句,Python 产生一个异常事件通知用户;
  • 在第 3 行,打印 BBB。

程序运行的结果如下:

AAA
Traceback (most recent call last):
  File "control-flow.py", line 2, in <module>
    100 / 0
ZeroDivisionError: division by zero
代码块12345
  • 在第 1 行,程序输出 AAA;
  • 在第 3 行,指明了产生异常的位置:File “control-flow.py”, line 2;在文件 “control-flow.py” 的第 2 行,产生了异常;这行信息非常重要,用于排查错误;
  • 在第 5 行,指明了异常的类型 ZeroDivisionError: division by zero;程序的执行流程被打断了,程序不再执行发生异常之后的代码;当发生异常时需要捕获处理它,否则程序会中止执行。

1.3 读取文件

编写一个读取文件内容的 Python 程序,如果不进行错误处理,代码如下:

file = open('test.txt')
line = file.readline()
print(line)
file.close()
代码块1234
  • 在第 1 行,打开文件 test.txt;
  • 在第 2 行,读取文件的一行;
  • 在第 3 行,打印;
  • 在第 4 行,关闭文件。

在下面的小节中,将使用异常处理对这个程序逐步进行改进。

2. try … except 语句

2.1 基本用法

Python 处理异常的基本语法如下:

try:
	可能发生异常的代码块
except:
	处理异常的代码块	
代码块1234
  • 在 try 关键字后,是可能发生异常的代码块;当发生异常后,程序跳转到处理异常的代码块;
  • 在 except 关键字后,是处理异常的代码块。

下面的程序首先抛出异常,然后捕获该异常,代码如下:

try:
    print('try:')
    100/0
    print('never reach here')
except:    
    print('except:')
代码块123456
  • 在第 2 行,打印字符串 ‘try:’;
  • 在第 3 行,执行 100/0,除数是 0,会抛出异常;
  • 在第 4 行,抛出异常后,程序跳转到处理异常的代码块,该行代码不会被执行;
  • 在第 6 行,捕获异常后,打印字符串 ‘except:’。

程序运行输出:

try:
except:
代码块12

2.2 处理指定类型的异常

在 except 关键字后加上异常类型,表示仅处理该类型的异常,语法如下:

try:
	可能发生异常的代码块
except 异常类型:
	处理异常的代码块	
代码块1234

下面的程序仅处理 ZeroDivisionError 类型的异常:

try:
    print('try:')
    100/0
    print('never reach here')
except ZeroDivisionError:
    print('except ZeroDivisionError:')
代码块123456
  • 在第 2 行,打印字符串 ‘try:’;
  • 在第 3 行,执行 100/0,除数是 0,会抛出 ZeroDivisionError 类型的异常;
  • 在第 4 行,抛出异常后,程序跳转到处理异常的代码块,该行代码不会被执行;
  • 在第 5 行,程序仅仅捕获 ZeroDivisionError 类型的异常;
  • 在第 6 行,捕获异常后,打印字符串 ‘except ZeroDivisionError:’。

程序运行输出:

2.3 处理多种类型的异常

可以使用多个 except 关键字处理多种类型的异常,语法如下:

try:
	可能发生异常的代码块
except 异常类型1:
	处理异常的代码块	
except 异常类型2:
	处理异常的代码块	
...		
代码块1234567

编写一个能够捕获两种类型的异常的程序,首先编写函数 generateError。函数 generateError 在运行时,可能抛出两种类型的异常,代码如下:

def generateError():
    import random
    number = random.randint(0, 1)
    if number == 0:
        100 / 0
    else:
        file = open('none-exsist-file')
代码块1234567
  • 在第 3 行,产生一个 [0, 1] 之间的随机数
  • 在第 4 行,如果随机数是 0在第 5 行,被除数是 0,产生 ZeroDivisionError 类型的异常
  • 在第 6 行,如果随机数是 1在第 7 行,打开一个不存在的文件,产生 IOError 类型的异常

编写捕获两种类型异常的程序:

try:
    print('try:')
    generateError()
    print('never reach here')
except ZeroDivisionError:
    print('except ZeroDivisionError:')
except IOError:    
    print('except IOError:')
代码块12345678
  • 在第 3 行,调用 generateError(),会随机抛出 ZeroDivisionError 类型或者 IOError 类型的异常;
  • 在第 5 行,程序捕获 ZeroDivisionError 类型的异常;在第 6 行,捕获异常后,打印字符串 ‘except ZeroDivisionError:’;
  • 在第 7 行,程序捕获 IOError 类型的异常;在第 8 行,捕获异常后,打印字符串 ‘except IOError:’。

2.4 except … as

在捕获异常时,不仅可以获取异常类型,还可以获取异常对象,语法如下:

except 异常类型 as 异常对象:
代码块1

下面的例子处理异常时,同时获取了异常类型和异常对象:

try:
	list = ['www', 'imooc', 'com']
	print(list[3])
except Exception as e:
	print('except: %s' % e)
代码块12345
  • 在第 4 行,异常类型为 Exception,异常对象为 e
  • 在第 5 行,打印异常对象 e

程序输出如下:

except: list index out of range
代码块1

2.5 读取文件

下面的程序实现 1.3 小节读取文件的功能需求:

try:
    file = open('test.txt')
    line = file.readline()
    print(line)
    file.close()
except IOError:    
    print('except IOError:')
代码块1234567
  • 在第 2 行,调用 open 函数可能会产生 IOError;
  • 在第 3 行,调用 readline 函数可能会产生 IOError;
  • 在第 5 行,关闭文件;当异常发生时,该行代码不会被执行
  • 在第 6 行,捕获 IOError 类型的异常。

这个版本的程序的缺陷在于,当异常发生时,关闭文件的代码不会被执行。文件打开后,没有及时关闭,会带来潜在的问题。在下面的小节中,将对这个程序进行改进。

3. try … else 语句

3.1 基本用法

在异常处理中 else 关键字用于指定没有异常时执行的代码块,语法如下:

try:
	可能发生异常的代码块
except:
	处理异常的代码块
else:
    没有异常时执行的代码块
代码块123456
  • 当发生异常时,执行 except 对应的代码块
  • 当没有发生异常时,执行 else 对应的代码块

3.2 读取文件

下面的程序实现 1.3 小节读取文件的功能需求:

try:
    file = open('test.txt')
    line = file.readline()
except IOError:    
    print('except IOError:')
else:
    print(line)
    file.close()    
代码块12345678
  • 在第 2 行,调用 open 函数可能会产生 IOError;
  • 在第 3 行,调用 readline 函数可能会产生 IOError;
  • 在第 5 行,关闭文件;当异常发生时,该行代码不会被执行
  • 在第 6 行,else 关键字定义了没有异常时执行的代码;在第 7 行,打印文件内容;在第 8 行,关闭文件。

这个版本的程序的仍然存在缺陷,当异常发生时,关闭文件的代码不会被执行。文件打开后,没有及时关闭,会带来潜在的问题。在下面的小节中,将对这个程序进行改进。

4. try … finally 语句

4.1 基本用法

在异常处理中,finally 关键字用于指定无论是否发生异常都需要执行的代码块,语法如下:

try:
	可能发生异常的代码块
except:
	处理异常的代码块
finally:
    无论是否发生异常都会执行的代码块
代码块123456

下面的程序在执行过程中没有异常:

try:
    print('try:')
finally:    
    print('finally:')
代码块1234

程序输出:

try:
finally:
代码块12

下面的程序在执行过程中产生异常:

try:
    print('try:')
    100 / 0
finally:    
    print('finally:')
代码块12345

程序输出:

try:
finally:
代码块12

可以看出,无论是否发生异常,finally 定义的代码块总是被执行

4.2 读取文件

下面的程序实现 1.3 小节读取文件的功能需求:

try:
    file = open('test.txt')
  	line = file.readline()
   	print(line)
except IOError:
    print('except IOError:')
finally:
	if file:
  		file.close()
代码块123456789
  • 在第 2 行,调用 open 函数可能会产生 IOError;如果在此处产生 IOError,变量 file 的值为空
  • 在第 3 行,调用 readline 函数可能会产生 IOError;如果在此处产生 IOError,因为已经成功打开了文件,变量 file 的值不为空;
  • 在第 5 行,捕获 IOError 类型的异常;
  • 在第 7 行,finally 关键字定义了最终需要执行的代码块;发生异常时,会执行该代码块;没有异常时,也会执行该代码块;
  • 在第 8 行,检查变量 file 的值是否为空;如果程序在 open 的地方发生异常,变量 file 的值为空,不需要关闭文件;如果程序在 readline 的地方发生异常,变量 file 的值不为空,需要关闭文件。

5. raise 语句

Python 提供了 raise 语句用于抛出异常,raise 语句有 3 种形式:

形式

功能

raise

不带任何参数

raise Exception

把异常的名称作为参数

raise Exception(info)

把异常的名称、异常的描述信息作为参数

5.1 raise

try:
	print('try:')
	raise
	print('never reach here')
except:
	print('except:')
代码块123456
  • 在第 3 行,使用 raise 抛出异常;
  • 在第 4 行,不会执行这行代码,执行 raise 后,程序流程跳转到第 5 行;
  • 在第 5 行,捕获程序抛出的异常。

程序输出如下:

try:
except:
代码块12

5.2 raise Exception

try:
	print('try:')
	raise ValueError
	print('never reach here')
except ValueError:
	print('except ValueError:')
代码块123456
  • 在第 3 行,使用 raise 抛出特定类型的异常 ValueError;
  • 在第 4 行,不会执行这行代码,执行 raise 后,程序流程跳转到第 5 行;
  • 在第 5 行,捕获程序抛出的 ValueError 类型的异常。

程序输出如下:

try:
except ValuseError:
代码块12

5.3 raise Exception(info)

编写程序 raise.py 如下:

try:
	text = input('Please input digit: ')
	if not text.isdigit():
		info = '"%s" is not digit' % text
		raise ValueError(info)
except ValueError as e:
	print('except ValueError: %s' % e)	
代码块1234567
  • 在第 2 行,提示用户输入数字;
  • 在第 3 行,如果用户输入的不是数字;在第 4 行,拼接字符串 info 用于描述错误的具体信息;在第 5 行,ValueError(info) 创建了一个对象,包括:异常类型和错误信息,使用 raise 抛出该异常对象;
  • 在第 6 行,捕获程序抛出的 ValueError 类型的异常,变量 e 指向 raise 语句抛出的异常。

程序输出如下:

C:\> python raise.py
Please input digit: abc
try:
except ValuseError: abc is not digit
代码块1234
  • 在第 2 行,用户输入 abc
  • 在第 4 行,提示用户的输入错误: “abc is not digit”,具体的错误信息对用户要友好

欢迎关注「慕课网」,发现更多IT圈优质内容,分享干货知识,帮助你成为更好的程序员!

相关推荐

笔记本电脑选哪个品牌比较好

1、苹果APPLE/美国2、戴尔DELL/美国3、华为HUAWEI/中国4、小米MI/中国5、微软Microsoft/美国6、联想LENOVO/中国7、惠普HP/美国8、华硕ASUS/...

10系列显卡排名(10系显卡性能排行)

十系显卡指NVIDIAGeForce10系列,是英伟达研发并推出的图形处理器系列,被用以取代NVIDIAGeForce900系列图形处理器。新系列采用帕斯卡微架构来代替之前的麦克斯韦微架构,并...

最新win7系统下载(windows7最新版本下载)
最新win7系统下载(windows7最新版本下载)

最简单的方法就是,下载完镜像文件后,直接把镜像文件解压,解压到非C盘,然后在解压文件里面找到setup.exe,点击运行即可。安装系统完成后,在C盘找到一个Windows.old(好几个GB,是旧系统打包在这里,垃圾文件了)删除即可。扩展资...

2026-01-15 06:43 off999

哪个电脑管家软件好用(哪个电脑管家好用些)

腾讯电脑管家吧,因为这个是杀毒和管理合一的,占用内存小,因此显得更为简洁,使电脑运行更加流畅此外电脑诊所,工具箱以及4+1的杀毒模式让腾讯电脑管家也收到了广泛的关注4+1杀毒引擎,管家反病毒引擎、金山...

怎么进入win7安全模式(怎么进入win7安全模式界面)

方法如下:1、首先进入Win7系统,然后使用Win键+R组合键打开运行框,输入“Msconfig”回车进入系统配置。2、在打开的系统配置中,找到“引导”选项,然后单击,选择Win7的引导项,然后在“安...

怎么分区固态硬盘(怎样分区固态硬盘)

固态硬盘的分区方法与传统机械硬盘基本相同,以下是一个简单的步骤:1.打开磁盘管理工具:在Windows操作系统中,按下Win+X键,选择"磁盘管理"。或者打开控制面板,在"...

笔记本声卡驱动怎么下载(笔记本如何下载声卡)
笔记本声卡驱动怎么下载(笔记本如何下载声卡)

1、在浏览器中输入并搜索,然后下载并安装。2、安装完成后打开360驱动大师,它就会自动检测你的电脑需要安装或升级的驱动。3、检测完毕后,我们可以看到我们的声卡驱动需要安装或升级,点击安装或升级,就会开始自动安装或升级声卡了。4、升级过程中会...

2026-01-15 05:43 off999

win10加快开机启动速度(加快开机速度 win10)

一、启用快速启动功能1.按win+r键调出“运行”在输入框输入“gpedit.msc”按回车调出“组策略编辑器”?2.在“本地组策略编辑器”依次打开“计算机配置——管理模块——系统——关机”在右侧...

excel的快捷键一览表(excel的快捷键一览表超全)
excel的快捷键一览表(excel的快捷键一览表超全)

Excel快捷键大全的一些操作如下我在工作中经常使用诸如word或Excel之类的办公软件。我相信每个人都不太熟悉这些办公软件的快捷键。使用快捷键将提高办公效率,并使您的工作更加轻松快捷。。例如,在复制时,请使用CtrI+C进行复制,...

2026-01-15 05:03 off999

华硕u盘启动按f几(华硕u盘装系统按f几进入)

F8。1、开机的同时按F8进入BIOS。2、在Boot菜单中,置secure为disabled。3、BootListOption置为UEFI。4、在1stBootPriority中usb—HD...

bootmgr(bootmgrismissing开机不了怎么办)
  • bootmgr(bootmgrismissing开机不了怎么办)
  • bootmgr(bootmgrismissing开机不了怎么办)
  • bootmgr(bootmgrismissing开机不了怎么办)
  • bootmgr(bootmgrismissing开机不了怎么办)
手机云电脑怎么用(手机云端电脑)

使用手机云电脑,您首先需要安装相应的云电脑应用。例如,华为云电脑APP。在安装并打开应用后,您将看到一个显示器的图标,这就是您的云电脑。点击这个图标,您将被连接到一个预装有Windows操作系统和必要...

ie11浏览器怎么安装(ie11浏览器安装步骤)

如果IE浏览器11版本你发现无法正常安装,那么很可能是这样几个原因,一个就是电脑的存储空间不够到时无法安装,再有就是网络的问题,如果没有办法安装的话就不要再安装了,本身这个IE浏览器并不是多好用,你最...

台式机重装系统win7(台式机怎么重装win7)

下面主要介绍两种方法以重装系统:一、U盘重装系统准备:一台正常开机的电脑和一个U盘1、百度下载“U大师”(老毛桃、大白菜也可以),把这个软件下载并安装在电脑上。2、插上U盘,选择一键制作U盘启动(制作...

字母下划线怎么打出来(字母下的下划线怎么去不掉)

第一步,在电脑上找到文字处理软件WPS,双击即自动新建一个新文档。第二步,在文档录入需要处理的字母和数字,双击鼠标或拖动鼠标选择要处理的内容。第三步,在页面的左上方的横向菜单栏,找到字母U的按纽,点击...

取消回复欢迎 发表评论: