Python学习笔记 异常处理 DAY3

异常处理

1.Python异常总结

  • 程序在运行过程当中,不可避免的会出现一些错误,比如:
    使用了没有赋值过的变量
    使用了不存在的索引
    除0

    这些错误在程序中,我们称其为异常。
    程序运行过程中,一旦出现异常将会导致程序立即终止,异常以后的代码全部都不会执行!

异常体系内部有层次关系,Python异常体系中的部分关系如下所示:

在这里插入图片描述

异常类的层次结构如下:

BaseException  # 所有异常的基类
+-- SystemExit  # 解释器请求退出
+-- KeyboardInterrupt  # 用户中断执行(通常是输入^C)
+-- GeneratorExit  # 生成器(generator)发生异常来通知退出
+-- Exception  # 常规异常的基类
+-- StopIteration  # 迭代器没有更多的值
+-- StopAsyncIteration  # 必须通过异步迭代器对象的__anext__()方法引发以停止迭代
+-- ArithmeticError  # 各种算术错误引发的内置异常的基类
  +-- FloatingPointError  # 浮点计算错误
  +-- OverflowError  # 数值运算结果太大无法表示
  +-- ZeroDivisionError  # 除(或取模)零 (所有数据类型)
+-- AssertionError  # 当assert语句失败时引发
+-- AttributeError  # 属性引用或赋值失败
+-- BufferError  # 无法执行与缓冲区相关的操作时引发
+-- EOFError  # 当input()函数在没有读取任何数据的情况下达到文件结束条件(EOF)时引发
+-- ImportError  # 导入模块/对象失败
  +-- ModuleNotFoundError  # 无法找到模块或在在sys.modules中找到None
+-- LookupError  # 映射或序列上使用的键或索引无效时引发的异常的基类
  +-- IndexError  # 序列中没有此索引(index)
  +-- KeyError  # 映射中没有这个键
+-- MemoryError  # 内存溢出错误(对于Python 解释器不是致命的)
+-- NameError  # 未声明/初始化对象 (没有属性)
  +-- UnboundLocalError  # 访问未初始化的本地变量
+-- OSError  # 操作系统错误,EnvironmentError,IOError,WindowsError,socket.error,select.error和mmap.error已合并到OSError中,构造函数可能返回子类
  +-- BlockingIOError  # 操作将阻塞对象(e.g. socket)设置为非阻塞操作
  +-- ChildProcessError  # 在子进程上的操作失败
  +-- ConnectionError  # 与连接相关的异常的基类
    +-- BrokenPipeError  # 另一端关闭时尝试写入管道或试图在已关闭写入的套接字上写入
    +-- ConnectionAbortedError  # 连接尝试被对等方中止
    +-- ConnectionRefusedError  # 连接尝试被对等方拒绝
    +-- ConnectionResetError  # 连接由对等方重置
  +-- FileExistsError  # 创建已存在的文件或目录
  +-- FileNotFoundError  # 请求不存在的文件或目录
  +-- InterruptedError  # 系统调用被输入信号中断
  +-- IsADirectoryError  # 在目录上请求文件操作(例如 os.remove())
  +-- NotADirectoryError  # 在不是目录的事物上请求目录操作(例如 os.listdir())
  +-- PermissionError  # 尝试在没有足够访问权限的情况下运行操作
  +-- ProcessLookupError  # 给定进程不存在
  +-- TimeoutError  # 系统函数在系统级别超时
+-- ReferenceError  # weakref.proxy()函数创建的弱引用试图访问已经垃圾回收了的对象
+-- RuntimeError  # 在检测到不属于任何其他类别的错误时触发
  +-- NotImplementedError  # 在用户定义的基类中,抽象方法要求派生类重写该方法或者正在开发的类指示仍然需要添加实际实现
  +-- RecursionError  # 解释器检测到超出最大递归深度
+-- SyntaxError  # Python 语法错误
  +-- IndentationError  # 缩进错误
  +-- TabError  # Tab和空格混用
+-- SystemError  # 解释器发现内部错误
+-- TypeError  # 操作或函数应用于不适当类型的对象
+-- ValueError  # 操作或函数接收到具有正确类型但值不合适的参数
  +-- UnicodeError  # 发生与Unicode相关的编码或解码错误
  +-- UnicodeDecodeError  # Unicode解码错误
  +-- UnicodeEncodeError  # Unicode编码错误
  +-- UnicodeTranslateError  # Unicode转码错误
+-- Warning  # 警告的基类
+-- DeprecationWarning  # 有关已弃用功能的警告的基类
+-- PendingDeprecationWarning  # 有关不推荐使用功能的警告的基类
+-- RuntimeWarning  # 有关可疑的运行时行为的警告的基类
+-- SyntaxWarning  # 关于可疑语法警告的基类
+-- UserWarning  # 用户代码生成警告的基类
+-- FutureWarning  # 有关已弃用功能的警告的基类
+-- ImportWarning  # 关于模块导入时可能出错的警告的基类
+-- UnicodeWarning  # 与Unicode相关的警告的基类
+-- BytesWarning  # 与bytes和bytearray相关的警告的基类
+-- ResourceWarning  # 与资源使用相关的警告的基类。被默认警告过滤器忽略。

以上内容并非原创,是借鉴了大牛的讲解,特此引用,方便学习。详情请参考:https://blog.csdn.net/polyhedronx/article/details/81589196

2.Python警告总结

  • Warning:警告的基类
  • DeprecationWarning:关于被弃用的特征的警告
  • FutureWarning:关于构造将来语义会有改变的警告
  • UserWarning:用户代码生成的警告
  • PendingDeprecationWarning:关于特性将会被废弃的警告
  • RuntimeWarning:可疑的运行时行为(runtime behavior)的警告
  • SyntaxWarning:可疑语法的警告
  • ImportWarning:用于在导入模块过程中触发的警告
  • UnicodeWarning:与Unicode相关的警告
  • BytesWarning:与字节或字节码相关的警告
  • ResourceWarning:与资源使用相关的警告

warning模块主要有两个方法:

1.warn

该方法用于输出一个警告信息 
#参数一: message 表示警告的详细信息
#参数二: category 类别,需要一个类作为参数,该类必须是Warning的子类,Warning类是Exception的子类,用于给警告设置一个具体的类型
#参数三: stacklevel 指定调用栈的层级,用于确定要从哪一级获取行号
#参数四: 待研究
# Code typically replaced by _warnings
def warn(message, category=None, stacklevel=1, source=None):
    """Issue a warning, or maybe ignore it or raise an exception."""
    # Check if message is already a Warning object
    if isinstance(message, Warning):
        category = message.__class__
    # Check category argument
    if category is None:
        category = UserWarning
    if not (isinstance(category, type) and issubclass(category, Warning)):
        raise TypeError("category must be a Warning subclass, "
                        "not '{:s}'".format(type(category).__name__))
    # Get context information
    try:
        if stacklevel <= 1 or _is_internal_frame(sys._getframe(1)):
            # If frame is too small to care or if the warning originated in
            # internal code, then do not try to hide any frames.
            frame = sys._getframe(stacklevel)
        else:
            frame = sys._getframe(1)
            # Look for one frame less since the above line starts us off.
            for x in range(stacklevel-1):
                frame = _next_external_frame(frame)
                if frame is None:
                    raise ValueError
    except ValueError:
        globals = sys.__dict__
        lineno = 1
    else:
        globals = frame.f_globals
        lineno = frame.f_lineno
    if '__name__' in globals:
        module = globals['__name__']
    else:
        module = "<string>"
    filename = globals.get('__file__')
    if filename:
        fnl = filename.lower()
        if fnl.endswith(".pyc"):
            filename = filename[:-1]
    else:
        if module == "__main__":
            try:
                filename = sys.argv[0]
            except AttributeError:
                # embedded interpreters don't have sys.argv, see bug #839151
                filename = '__main__'
        if not filename:
            filename = module
    registry = globals.setdefault("__warningregistry__", {})
    warn_explicit(message, category, filename, lineno, module, registry,
                  globals, source)

以上为源码中的warn方法。

注:在查看源码是,发现了一个用于表示警告消息的类

class WarningMessage(object):

    _WARNING_DETAILS = ("message", "category", "filename", "lineno", "file",
                        "line", "source")

    def __init__(self, message, category, filename, lineno, file=None,
                 line=None, source=None):
        self.message = message
        self.category = category
        self.filename = filename
        self.lineno = lineno
        self.file = file
        self.line = line
        self.source = source
        self._category_name = category.__name__ if category else None

    def __str__(self):
        return ("{message : %r, category : %r, filename : %r, lineno : %s, "
                    "line : %r}" % (self.message, self._category_name,
                                    self.filename, self.lineno, self.line))

该类非常简单,就是保存了一些警告信息,然后覆盖了 __str__函数,用于自定义字符串形式。

3.处理、抛出异常的方法

程序运行时出现异常,目的并不是让我们的程序直接终止!
Python是希望在出现异常时,我们可以编写代码来对异常进行处理!

try语句
try:
代码块(可能出现错误的语句)
except 异常类型 as 异常名:
代码块(出现错误以后的处理方式)
except 异常类型 as 异常名:
代码块(出现错误以后的处理方式)
except 异常类型 as 异常名:
代码块(出现错误以后的处理方式)
else:
代码块(没出错时要执行的语句)
finally:
代码块(该代码块总会执行)

try是必须的 else语句有没有都行,except和finally至少有一个 。

可以将可能出错的代码放入到try语句,这样如果代码没有错误,则会正常执行,
如果出现错误,则会执行expect子句中的代码,这样我们就可以通过代码来处理异常
避免因为一个异常导致整个程序的终止

try - except 语句

  • 将try - except、try - except - else、try - except - finally整理在一起。
# print('hello')
# try:
#     # try中放置的是有可能出现错误的代码
#     print(10/0)
# except:
#     # except中放置的是出错以后的处理防暑
#     print('哈哈哈,出错了~~~')
# else:
#     print('程序正常执行没有错误')
# print('你好')
# print(10/0)
def fn():
    print('Hello fn')
    print(a)
    print(10/0)

def fn2():
    print('Hello fn2')
    fn()

def fn3():
    print('Hello fn3')
    fn2()

fn3()
# NameError: name 'a' is not defined

抛出异常

  • 当在函数中出现异常时,如果在函数中对异常进行了处理,则异常不会再继续传播,
    如果函数中没有对异常进行处理,则异常会继续向函数调用处传播,
    如果函数调用处处理了异常,则不再传播,如果没有处理则继续向调用处传播
    直到传递到全局作用域(主模块)如果依然没有处理,则程序终止,并且显示异常信息

  • 当程序运行过程中出现异常以后,所有的异常信息会被保存一个专门的异常对象中,
    而异常传播时,实际上就是异常对象抛给了调用处
    比如 : ZeroDivisionError类的对象专门用来表示除0的异常
    NameError类的对象专门用来处理变量错误的异常

Python为我们提供了多个异常对象 。

print('异常出现前')
l = []
try:
    # print(c)
    # l[10]
    # 1 + 'hello'
    print(10/0)
except NameError:
    # 如果except后不跟任何的内容,则此时它会捕获到所有的异常
    # 如果在except后跟着一个异常的类型,那么此时它只会捕获该类型的异常
    print('出现 NameError 异常')
except ZeroDivisionError:
    print('出现 ZeroDivisionError 异常')
except IndexError:
    print('出现 IndexError 异常')
# Exception 是所有异常类的父类,所以如果except后跟的是Exception,他也会捕获到所有的异常
# 可以在异常类后边跟着一个 as xx 此时xx就是异常对象
except Exception as e :
    print('未知异常',e,type(e))
finally :
    print('无论是否出现异常,该子句都会执行')

print('异常出现后')

raise语句抛出异常

  • 可以使用 raise 语句来抛出异常,
    raise语句后需要跟一个异常类 或 异常的实例
# 也可以自定义异常类,只需要创建一个类继承Exception即可
class MyError(Exception):
    pass

def add(a,b):
    # 如果a和b中有负数,就向调用处抛出异常
    if a < 0 or b < 0:
        # raise用于向外部抛出异常,后边可以跟一个异常类,或异常类的实例
        # raise Exception    
        # 抛出异常的目的,告诉调用者这里调用时出现问题,希望你自己处理一下
        # raise Exception('两个参数中不能有负数!')  
        raise MyError('自定义的异常')
        
        # 也可以通过if else来代替异常的处理
        # return None
    r = a + b
    return r

print(add(-123,456))    

4.练习题

1、猜数字游戏

题目描述:

电脑产生一个零到100之间的随机数字,然后让用户来猜,如果用户猜的数字比这个数字大,提示太大,否则提示太小,当用户正好猜中电脑会提示,“恭喜你猜到了这个数是…”。在用户每次猜测之前程序会输出用户是第几次猜测,如果用户输入的根本不是一个数字,程序会告诉用户"输入无效"。

(尝试使用try catch异常处理结构对输入情况进行处理)

获取随机数采用random模块。

答:

import random
count = 1
IntegerNum = random.randint(0,100)
while count > 0 :
    try:
        num = int(input("第"+str(count)+"次猜,请输入一个数字:"))
        if num == IntegerNum:
            print("恭喜你猜对了,这个数是",IntegerNum)
            break
        elif num > IntegerNum:
            count += 1
            print("太大")
        else:
            count += 1
            print("太小")
    except Exception as e :
        print("输入无效")
        count += 1
# 第1次猜,请输入一个数字:adsfhjk
# 输入无效
# 第2次猜,请输入一个数字:27
# 太大
# 第3次猜,请输入一个数字:15
# 太小
# 第4次猜,请输入一个数字:21
# 太大
# 第5次猜,请输入一个数字:17
# 太小
# 第6次猜,请输入一个数字:
# 输入无效
# 第7次猜,请输入一个数字:19
# 太小
# 第8次猜,请输入一个数字:20
# 恭喜你猜对了,这个数是 20
     print("太大")
        else:
            count += 1
            print("太小")
    except Exception as e :
        print("输入无效")
        count += 1
# 第1次猜,请输入一个数字:adsfhjk
# 输入无效
# 第2次猜,请输入一个数字:27
# 太大
# 第3次猜,请输入一个数字:15
# 太小
# 第4次猜,请输入一个数字:21
# 太大
# 第5次猜,请输入一个数字:17
# 太小
# 第6次猜,请输入一个数字:
# 输入无效
# 第7次猜,请输入一个数字:19
# 太小
# 第8次猜,请输入一个数字:20
# 恭喜你猜对了,这个数是 20
Python中,异常处理是非常重要的一部分。当程序运行时如果出现错误,如果没有异常处理,程序就会崩溃。为了避免这种情况,Python提供了异常处理机制。 在Python中,异常处理语句使用 `try` 和 `except` 关键字来实现。`try` 语句块中包含可能会发生异常的代码,如果这段代码出现了异常,则会跳转到 `except` 语句块中执行异常处理代码。 下面是一个简单的例子: ```python try: num = int(input("输入一个整数:")) print(10/num) except ZeroDivisionError: print("除数不能为0") except ValueError: print("输入的不是整数") ``` 在上面的代码中,我们尝试将用户输入的字符串转换为整数,并将其用作除数计算 10/num。如果用户输入的是 0,则会触发 ZeroDivisionError 异常。如果用户输入的不是整数,则会触发 ValueError 异常。如果发生异常,则会跳转到对应的 except 语句块中执行处理代码。 除了可以指定具体的异常类型,也可以使用 `except Exception` 来捕获所有异常。例如: ```python try: num = int(input("输入一个整数:")) print(10/num) except Exception as e: print("发生异常:", e) ``` 在上面的代码中,如果发生任何异常,都会跳转到 `except` 语句块中执行处理代码,并将异常信息打印出来。 除了 `try` 和 `except`,还有 `finally` 关键字,它指定的代码块无论是否发生异常都会执行。例如: ```python try: num = int(input("输入一个整数:")) print(10/num) except Exception as e: print("发生异常:", e) finally: print("程序执行完毕") ``` 在上面的代码中,无论是否发生异常,都会执行 `finally` 中的代码,即输出“程序执行完毕”。 之,在Python中,异常处理是非常重要的一部分,它可以有效避免程序崩溃,提高程序的健壮性和可靠性。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

沧浪之水、

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值