这里介绍Python中的异常
data:image/s3,"s3://crabby-images/96f1b/96f1b3e681ba8ae4eea8de3792d5b02aab4c1d28" alt="abstract.png"
自定义异常
Python中异常的基类为BaseException。其常见子类有:SystemExit、KeyboardInterrupt、Exception等。其中,Exception是常规异常的基类。当我们自定义异常类时,建议直接或间接继承Exception类。而不是直接继承BaseException
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42
| class CustomException(Exception): pass
class RuntimeException(ZeroDivisionError): pass
class BusinessException(Exception): def __init__(self, code, msg): super().__init__(self) self.code = code self.msg = msg
def __str__(self): detail = { "code": self.code, "msg": self.msg, } return str(detail) def get_code(self): return self.code def get_msg(self): return self.msg
if __name__ == "__main__": try: raise BusinessException(520, "参数校验失败") except BusinessException as ex: error_code = ex.get_code() error_msg = ex.get_msg() print(f"error code : {error_code}; error msg : {error_msg}") print("happen business exception:", ex) except Exception as ex: print("ex:", ex)
|
data:image/s3,"s3://crabby-images/5a85e/5a85e78090c9801c6d6ecbc68dcf4e6516a5e61c" alt="figure 0.jpg"
捕获异常
可以通过except子句捕获指定异常。具体地,可以在一个except子句中使用元组指定多个异常;也可以同时使用多个excpet子句
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23
| print("-------------- 捕获异常 -------------") try: num = 5/0 except ZeroDivisionError: print("[#1] 除数不能为0")
print("\n\n-------------- 捕获多个异常 -------------") for i in range(0,2) : try: if( i==1 ): i = "aaron" answer = 5 / i except (ZeroDivisionError, SyntaxError) as ex: print("Error: 除数为0, ex:", ex) except TypeError: print("发生语法错误") except : print("发生异常")
|
data:image/s3,"s3://crabby-images/d8ff1/d8ff1a1139b3fc6c9797e77e36e5ac1f2d17d8df" alt="figure 1.png"
try-except的finally子句: 无论try中发生任何异常,finally中的代码都会被执行。即使try中发生的异常未被捕获,finally中代码在全部执行完成后,会再次重新抛出相应异常
data:image/s3,"s3://crabby-images/77bd6/77bd66498d95bd899df8016f8858c87fc93d9675" alt="figure 2.png"
抛异常
1. 基本语法
可通过raise子句来抛出异常
1 2
| raise AttributeError("用户姓名重复")
|
data:image/s3,"s3://crabby-images/29c5c/29c5c4bb924131d2410b99a626de305d5ad01b17" alt="figure 3.png"
2. 异常捕获后继续抛该异常
捕获异常后,可直接使用raise,其会继续抛出当前捕获的异常
1 2 3 4 5 6
| try: num = 5/0 except ZeroDivisionError: print("[#1] 除数不能为0") raise
|
data:image/s3,"s3://crabby-images/22efc/22efcf0e8415f7db7e84e6e8cc8913528ca23695" alt="figure 4.png"
3. 异常捕获后抛其他异常
捕获异常后,可使用raise抛出其他的异常。此时,导致进入except子句的异常将被作为异常上下文存储起来,并出现在最终的错误消息中
1 2 3 4 5 6 7
| try: num = 5/0 except ZeroDivisionError: print("[#2] 除数不能为0") raise OSError
|
data:image/s3,"s3://crabby-images/acd59/acd59ca551033de94dd63a4d8766558e596e2a28" alt="figure 5.png"
4. 异常捕获后抛其他异常,并重设异常上下文
捕获异常后,可使用raise抛出别的异常。同时,使用raise … from … 语句提供指定的异常上下文
1 2 3 4 5 6 7
| try: num = 5/0 except ZeroDivisionError: print("[#3] 除数不能为0") raise OSError from SyntaxError
|
data:image/s3,"s3://crabby-images/7afb2/7afb2a3c847918ddb4e97e411a70f622b58a6f64" alt="figure 6.png"
5. 异常捕获后抛其他异常,并清除异常上下文
捕获异常后,可使用raise抛出别的异常。同时,使用raise … from None 语句 清除 异常上下文
1 2 3 4 5 6 7
| try: num = 5/0 except ZeroDivisionError: print("[#4] 除数不能为0") raise OSError from None
|
data:image/s3,"s3://crabby-images/921ff/921fff561d6a3ab515fe2af01b4b2d7e28c755c1" alt="figure 7.png"
其他特性
else子句
try-except支持else子句。仅当try代码块成功执行,未抛出异常,才会执行else代码块
1 2 3 4 5 6 7 8
| for i in range(0,3) : try: print(f"i --->>> {i}") answer = 5 / i except ZeroDivisionError: print("[Error]: 除数i是0") else: print("answer:", answer)
|
data:image/s3,"s3://crabby-images/03981/03981cec3fc001a45768696b4ebfdf002e895f4b" alt="figure 8.png"
try-with语句
try-with语句封装了try-except-finally 范式。使得对于需要关闭的资源,不再需要显式地通过finllay子句进行关闭
1 2 3 4 5 6
| with open("main.txt") as file1, open("readme.txt") as file2: src = file1.read(60)
print(f"文件file1是否被关闭: {file1.closed}") print(f"文件file2是否被关闭: {file2.closed}")
|
data:image/s3,"s3://crabby-images/9b2cb/9b2cb853422d19865057aba7c979942999f4183c" alt="figure 9.png"
参考文献
- Python编程·第3版:从入门到实践 Eric Matthes著
- Python基础教程·第3版 Magnus Lie Hetland著
- 流畅的Python·第1版 Luciano Ramalho著