06_Python异常处理机制
阅读原文时间:2023年07月09日阅读:1

1.异常概述

1.什么是错误: 错误是指有逻辑或语法等导致一个程序无法正常执行的问题
    2.什么是异常: 异常时程序出错时标识的一种状态,程序不会向下执行而转去调用此函数的地方等待处理错误并恢复为正常状态

    3.异常的作用: 通知上层调用者有错误产生需要处理,用做信号

    4.为什么要有异常处理机制: 在程序调用层数较深时,向主调函数传递错误信息需要层层的返回比较麻烦,所以用异常处理机制

2.异常处理语法

try-except语法: 尝试捕获异常,将程序转为正常状态并继续执行

        try:

            可能触发异常的语句

        except 错误类型1 as 变量1:

            异常处理语句1

        except 错误类型2 as 变量2:

            异常处理语句2

        else:

            未发生异常时会执行的语句

        finally:

            无论有无异常都会执行的语句

    try-finally语法: finally子句不能省略且不存在except子句,不会改变程序的(正常/异常)状态

        try:

            可能触发异常的语句

        finally:

            无论有无异常都会执行的语句

3.完整的异常语法示例

try:

        # 提示用户输入一个整数

        num = int(input("请输入一个整数:"))

        # 使用8除以用户输入的整数并且输出

        print("8除以%d的结果是:%.2f" % (num, 8/num))

    except ZeroDivisionError:

        print("除0错误")

    except ValueError:

        print("请输入正确的正整数")

    except Exception as e:

        print("未知错误:%s" % e)

    else:

        print("尝试成功")

    finally:

        print("无论有无异常发生必须执行此句")

4.主动抛出异常语句raise

作用: 触发一个错误让程序进入异常状态

    语法:

        raise 异常类型

        或 raise 异常对象

    示例:

        # 主动抛出异常

        def input_password():

            # 提示用户输入密码

            pwd = input("请输入密码:")

            # 判断密码长度>=8,返回用户输入的密码

            if len(pwd) >= 8:

                return pwd

            # 如果<8主动抛出异常

            print("主动抛出异常")

            # 创建一个异常对象(可以使用错误字符串信息作为参数)

            ex = Exception("密码长度不够")

            # raise主动抛出异常

            raise ex

try:

            print(input_password())

        except Exception as e:

            print(e)

5.异常的传递特性

def fun1():

        return int(input("请输入整数: "))

def fun2():

        return fun1()

# 利用异常传递性在主程序捕获异常

    try:

        print(fun2())

    except Exception as e:

        print("未知错误: %s" % e)

6.自定义异常

class MsgException(BaseException):

        def __init__(self, msg):

            self.msg = msg

try:

        raise MsgException("消息类型错误")

    except MsgException as e:

        print(e)

7.断言语句assert

作用: 当真值表达式为False时用错误数据创建一个AssertionError类型的错误并进入异常状态

    语法

        assert 真值表达式

    等同于

        if 真值表达式 == False:

            raise AssertionError(错误数据)

    示例1:

        # assert断言会终止程序并抛出异常

        def test1():

            print("模拟大量逻辑处理")

            ret = 100

            return ret

ret = test1()

        assert ret == 1000, "ret变量未使用到"

        ret1 = ret + 100

        print(ret1)

        # 执行结果: AssertionError: ret变量未使用到

    示例2:

        def get_age():

            age = input("请输入年龄: ")

            age = int(age)

            assert age < 140, "年龄不能大于140岁!"

            assert age >= 0, "年龄不能为负数!"

            return age

try:

            age = get_age()

        except AssertionError as err:

            print("发生了断言错误,错误对象是: %s" % err)

            age = 0

        print("您输入的年龄是: %s" % age)

8.异常处理的高级用法-环境管理器(上下文管理器)

with语句实现上下文管理器概述:

        1.用于对资源访问的场合,确保使用过程中不管是否发生异常,都会执行必要有"清理"操作,并释放资源

        2.with语句与try-finally相似,并不会改变异常状态,常运用于文件打开后自动关闭,线程中锁的自动获取和释放

        2.语法: with 表达式1 [as 变量名1], 表达式2 [as 变量名2], …  # as 子句用于绑定表达式创建的对象

    示例:

        # 打开文件读取文件数据(with来实现关闭文件)

        def read_file():

            try:

                # f = open("abcd.txt")

                with open('abcd.txt') as f:

                    while True:

                    s = f.readline()

                    if not s:

                        break

                    int(input("请输入任意数字打印下一行: "))

                    print(s)

                    print("文件已经关闭")

except IOError:

                print("出现异常已经捕获!")

            except ValueError:

                print("程序已转为正常状态")

read_file()

        print("程序结束")

9.环境管理器(上下文管理器)原理剖析

1.内部有__enter__ 和 __exit__方法的类被称为环境管理器,能够用with进行管理的对象必须是环境管理器

    2.__enter__ 将在进入with语句时被调用,并返回由as变量管理的对象

    3.__exit__ 将在离开with语句时被调用,且可以用参数来判断离开with语句时是否出现异常并做出相应的处理

class FileWriter:
def __init__(self, filename):
self.filename = filename # 此属性用于记住文件名

def writeline(self, s):  
    '''此方法用于向文件内写入字符串,同时自动添加换行'''  
    self.file.write(s)  
    self.file.write('\\n')

def \_\_enter\_\_(self):  
    '''此方法用于实现环境管理器'''  
    self.file = open(self.filename, 'w')  
    print("已进入\_\_enter\_\_方法,文件打开成功")  
    return self  # 返回值向用于 with中的as 绑定

def \_\_exit\_\_(self, exec\_type, exec\_value, exec\_tb):  
    '''  
    exec\_type: 为异常类异,没有异常发生时为None  
    exec\_value: 为错误的对象,没有异常时为None  
    exec\_tb: 为错误的traceback对象  
    '''  
    self.file.close()  
    print("文件", self.filename, "已经关闭")  
    if exec\_type is None:  
        print("退出with时没有发生异常")  
    else:  
        print("退出with时,有异常,类型是", exec\_type, "错误是", exec\_value)  
    print("\_\_exit\_\_法被调用,已离开with语句")

try:
with FileWriter("log.txt") as fw:
while True:
s = input("请输入一行: ")
if s == "exit":
break
if s == "error":
raise ValueError("故意制造的值错误")
fw.writeline(s)
except:
print("有错误发生,已转为正常")

print("这是with语句之外,也是程序的最后一条语句")

自定义的类实现环境管理器示例:

10.全部的异常类型

错误类型

ZeroDivisionError

除(或取模)零 (所有数据类型)

ValueError

传入无效的参数

AssertionError

断言语句失败

StopIteration

迭代器没有更多的值

IndexError

序列中没有此索引(index)

IndentationError

缩进错误

IOError

输入/输出操作失败

ImportError

导入模块/对象失败

NameError

未声明/初始化对象 (没有属性)

AttributeError

对象没有这个属性

 BaseException

 所有异常的基类

 SystemExit

 解释器请求退出

 KeyboardInterrupt

 用户中断执行(通常是输入^C)

 Exception

 常规错误的基类

 GeneratorExit

 生成器(generator)发生异常来通知退出

 StandardError

 所有的内建标准异常的基类

 ArithmeticError

 所有数值计算错误的基类

 FloatingPointError

 浮点计算错误

 OverflowError

 数值运算超出最大限制

 EOFError

 没有内建输入,到达EOF 标记

 EnvironmentError

 操作系统错误的基类

 OSError

 操作系统错误

 WindowsError

 系统调用失败

 LookupError

 无效数据查询的基类

 KeyError

 映射中没有这个键

 MemoryError

 内存溢出错误(对于Python 解释器不是致命的)

 UnboundLocalError

 访问未初始化的本地变量

 ReferenceError

  弱引用(Weak reference)试图访问已经垃圾回收了的对象  

 RuntimeError

 一般的运行时错误

 NotImplementedError

 尚未实现的方法

 SyntaxError

 Python  语法错误

 TabError

 Tab 和空格混用

 SystemError

 一般的解释器系统错误

 TypeError

 对类型无效的操作

 UnicodeError

 Unicode 相关的错误

 UnicodeDecodeError

 Unicode 解码时的错误

 UnicodeEncodeError

 Unicode 编码时错误

  UnicodeTranslateError  

 Unicode 转换时错误

警告类型

Warning

警告的基类

DeprecationWarning

关于被弃用的特征的警告

FutureWarning

关于构造将来语义会有改变的警告

OverflowWarning

旧的关于自动提升为长整型(long)的警告

  PendingDeprecationWarning  

关于特性将会被废弃的警告

RuntimeWarning

  可疑的运行时行为(runtime behavior)的警告  

SyntaxWarning

可疑的语法的警告

UserWarning

用户代码生成的警告