• python异常处理


    一、什么是程序异常

    在程序运行过程中,总会遇到各种各样的错误。

    有的错误是程序编写有问题造成的,比如本来应该输出整数结果输出了字符串,这种错误我们通常称之为bug,bug是必须修复的。

    有的错误是用户输入造成的,比如让用户输入email地址,结果得到一个空字符串,这种错误可以通过检查用户输入来做相应的处理。

    还有一类错误是完全无法在程序运行过程中预测的,比如写入文件的时候,磁盘满了,写不进去了,或者从网络抓取数据,网络突然断掉了。这类错误也称为异常,在程序中通常是必须处理的,否则,程序会因为各种问题终止并退出。

    比如我们执行下面的程序

    1. if __name__ == '__main__':
    2. a = 10/0
    3. print(a)

    执行结果

     

    可以看到执行没有成功而是发生了问题,这就是程序异常

    二、异常处理机制

    高级语言通常都内置了一套try...except...finally...的错误处理机制,Python也不例外。

    比如下面的程序

    1. try:
    2. print('try...')
    3. r = 10 / 0
    4. print('result:', r)
    5. except ZeroDivisionError as e:
    6. print('except:', e)
    7. finally:
    8. print('finally...')
    9. print('END')

    当我们认为某些代码可能会出错时,就可以用try来运行这段代码,如果执行出错,则后续代码不会继续执行,而是直接跳转至错误处理代码,即except语句块,执行完except后,如果有finally语句块,则执行finally语句块,至此,执行完毕。

    三、异常格式

    异常处理格式1:

    1. try:
    2. # 逻辑代码
    3. except Exception as e:
    4. # try中的代码如果有异常,则此代码块中的代码会执行。

    异常处理格式2: 

    1. try:
    2. # 逻辑代码
    3. except Exception as e:
    4. # try中的代码如果有异常,则此代码块中的代码会执行。
    5. finally:
    6. # try中的代码无论是否报错,finally中的代码都会执行,一般用于释放资源。
    7. print("end")

    特殊的finally

    当在函数或方法中定义异常处理的代码时,要特别注意finally和return。

    1. def func():
    2. try:
    3. return 123
    4. except Exception as e:
    5. pass
    6. finally:
    7. print(666)
    8. func()

    在try或except中即使定义了return,也会执行最后的finally块中的代码。 

    四、异常细分

    如果错误有很多种类,发生了不同类型的错误,之前只是简单的捕获了异常,出现异常则统一提示信息即可。如果想要对异常进行更加细致的异常处理,应该由不同的except语句块处理:

    1. try:
    2. print('try...')
    3. r = 10 / int('a')
    4. print('result:', r)
    5. except ValueError as e:
    6. print('ValueError:', e)
    7. except ZeroDivisionError as e:
    8. print('ZeroDivisionError:', e)
    9. finally:
    10. print('finally...')
    11. print('END')

    int()函数可能会抛出ValueError,所以我们用一个except捕获ValueError,用另一个except捕获ZeroDivisionError

    此外,如果没有错误发生,可以在except语句块后面加一个else,当没有错误发生时,会自动执行else语句:

    1. try:
    2. print('try...')
    3. r = 10 / int('2')
    4. print('result:', r)
    5. except ValueError as e:
    6. print('ValueError:', e)
    7. except ZeroDivisionError as e:
    8. print('ZeroDivisionError:', e)
    9. else:
    10. print('no error!')
    11. finally:
    12. print('finally...')
    13. print('END')

    如果想要对错误进行细分的处理,例如:发生Key错误和发生Value错误分开处理。

    基本格式:

    1. try:
    2. # 逻辑代码
    3. pass
    4. except KeyError as e:
    5. # 小兵,只捕获try代码中发现了键不存在的异常,例如:去字典 info_dict["n1"] 中获取数据时,键不存在。
    6. print("KeyError")
    7. except ValueError as e:
    8. # 小兵,只捕获try代码中发现了值相关错误,例如:把字符串转整型 int("无诶器")
    9. print("ValueError")
    10. except Exception as e:
    11. # 王者,处理上面except捕获不了的错误(可以捕获所有的错误)。
    12. print("Exception")

    Python中内置了很多细分的错误,供你选择。

    1. 常见异常:
    2. """
    3. AttributeError 试图访问一个对象没有的树形,比如foo.x,但是foo没有属性x
    4. IOError 输入/输出异常;基本上是无法打开文件
    5. ImportError 无法引入模块或包;基本上是路径问题或名称错误
    6. IndentationError 语法错误(的子类) ;代码没有正确对齐
    7. IndexError 下标索引超出序列边界,比如当x只有三个元素,却试图访问n x[5]
    8. KeyError 试图访问字典里不存在的键 inf['xx']
    9. KeyboardInterrupt Ctrl+C被按下
    10. NameError 使用一个还未被赋予对象的变量
    11. SyntaxError Python代码非法,代码不能编译(个人认为这是语法错误,写错了)
    12. TypeError 传入对象类型与要求的不符合
    13. UnboundLocalError 试图访问一个还未被设置的局部变量,基本上是由于另有一个同名的全局变量,
    14. 导致你以为正在访问它
    15. ValueError 传入一个调用者不期望的值,即使值的类型是正确的
    16. """
    17. 更多异常:
    18. """
    19. ArithmeticError
    20. AssertionError
    21. AttributeError
    22. BaseException
    23. BufferError
    24. BytesWarning
    25. DeprecationWarning
    26. EnvironmentError
    27. EOFError
    28. Exception
    29. FloatingPointError
    30. FutureWarning
    31. GeneratorExit
    32. ImportError
    33. ImportWarning
    34. IndentationError
    35. IndexError
    36. IOError
    37. KeyboardInterrupt
    38. KeyError
    39. LookupError
    40. MemoryError
    41. NameError
    42. NotImplementedError
    43. OSError
    44. OverflowError
    45. PendingDeprecationWarning
    46. ReferenceError
    47. RuntimeError
    48. RuntimeWarning
    49. StandardError
    50. StopIteration
    51. SyntaxError
    52. SyntaxWarning
    53. SystemError
    54. SystemExit
    55. TabError
    56. TypeError
    57. UnboundLocalError
    58. UnicodeDecodeError
    59. UnicodeEncodeError
    60. UnicodeError
    61. UnicodeTranslateError
    62. UnicodeWarning
    63. UserWarning
    64. ValueError
    65. Warning
    66. ZeroDivisionError
    67. """

     Python的错误其实也是class,所有的错误类型都继承自BaseException,所以在使用except时需要注意的是,它不但捕获该类型的错误,还把其子类也“一网打尽”。比如:

    1. try:
    2. foo()
    3. except ValueError as e:
    4. print('ValueError')
    5. except UnicodeError as e:
    6. print('UnicodeError')

    第二个except永远也捕获不到UnicodeError,因为UnicodeErrorValueError的子类,如果有,也被第一个except给捕获了。

    Python所有的错误都是从BaseException类派生的,常见的错误类型和继承关系

    1. BaseException
    2. ├── BaseExceptionGroup
    3. ├── GeneratorExit
    4. ├── KeyboardInterrupt
    5. ├── SystemExit
    6. └── Exception
    7. ├── ArithmeticError
    8. │ ├── FloatingPointError
    9. │ ├── OverflowError
    10. │ └── ZeroDivisionError
    11. ├── AssertionError
    12. ├── AttributeError
    13. ├── BufferError
    14. ├── EOFError
    15. ├── ExceptionGroup [BaseExceptionGroup]
    16. ├── ImportError
    17. │ └── ModuleNotFoundError
    18. ├── LookupError
    19. │ ├── IndexError
    20. │ └── KeyError
    21. ├── MemoryError
    22. ├── NameError
    23. │ └── UnboundLocalError
    24. ├── OSError
    25. │ ├── BlockingIOError
    26. │ ├── ChildProcessError
    27. │ ├── ConnectionError
    28. │ │ ├── BrokenPipeError
    29. │ │ ├── ConnectionAbortedError
    30. │ │ ├── ConnectionRefusedError
    31. │ │ └── ConnectionResetError
    32. │ ├── FileExistsError
    33. │ ├── FileNotFoundError
    34. │ ├── InterruptedError
    35. │ ├── IsADirectoryError
    36. │ ├── NotADirectoryError
    37. │ ├── PermissionError
    38. │ ├── ProcessLookupError
    39. │ └── TimeoutError
    40. ├── ReferenceError
    41. ├── RuntimeError
    42. │ ├── NotImplementedError
    43. │ └── RecursionError
    44. ├── StopAsyncIteration
    45. ├── StopIteration
    46. ├── SyntaxError
    47. │ └── IndentationError
    48. │ └── TabError
    49. ├── SystemError
    50. ├── TypeError
    51. ├── ValueError
    52. │ └── UnicodeError
    53. │ ├── UnicodeDecodeError
    54. │ ├── UnicodeEncodeError
    55. │ └── UnicodeTranslateError
    56. └── Warning
    57. ├── BytesWarning
    58. ├── DeprecationWarning
    59. ├── EncodingWarning
    60. ├── FutureWarning
    61. ├── ImportWarning
    62. ├── PendingDeprecationWarning
    63. ├── ResourceWarning
    64. ├── RuntimeWarning
    65. ├── SyntaxWarning
    66. ├── UnicodeWarning
    67. └── UserWarning

    使用try...except捕获错误还有一个巨大的好处,就是可以跨越多层调用,比如函数main()调用bar()bar()调用foo(),结果foo()出错了,这时,只要main()捕获到了,就可以处理:

    1. def foo(s):
    2. return 10 / int(s)
    3. def bar(s):
    4. return foo(s) * 2
    5. def main():
    6. try:
    7. bar('0')
    8. except Exception as e:
    9. print('Error:', e)
    10. finally:
    11. print('finally...')

    也就是说,不需要在每个可能出错的地方去捕获错误,只要在合适的层次去捕获错误就可以了。这样一来,就大大减少了写try...except...finally的麻烦。

    五、调用栈

    如果错误没有被捕获,它就会一直往上抛,最后被Python解释器捕获,打印一个错误信息,然后程序退出。来看看err.py

    1. # err.py:
    2. def foo(s):
    3. return 10 / int(s)
    4. def bar(s):
    5. return foo(s) * 2
    6. def main():
    7. bar('0')
    8. main()

    执行,结果如下:

    1. $ python3 err.py
    2. Traceback (most recent call last):
    3. File "err.py", line 11, in <module>
    4. main()
    5. File "err.py", line 9, in main
    6. bar('0')
    7. File "err.py", line 6, in bar
    8. return foo(s) * 2
    9. File "err.py", line 3, in foo
    10. return 10 / int(s)
    11. ZeroDivisionError: division by zero

    出错并不可怕,可怕的是不知道哪里出错了。解读错误信息是定位错误的关键。我们从上往下可以看到整个错误的调用函数链:

    错误信息第1行:

    Traceback (most recent call last):
    

    告诉我们这是错误的跟踪信息。

    第2~3行:

    1. File "err.py", line 11, in <module>
    2. main()

    调用main()出错了,在代码文件err.py的第11行代码,但原因是第9行:

    1. File "err.py", line 9, in main
    2. bar('0')

    调用bar('0')出错了,在代码文件err.py的第9行代码,但原因是第6行:

    1. File "err.py", line 6, in bar
    2. return foo(s) * 2

    原因是return foo(s) * 2这个语句出错了,但这还不是最终原因,继续往下看:

    1. File "err.py", line 3, in foo
    2. return 10 / int(s)

    原因是return 10 / int(s)这个语句出错了,这是错误产生的源头,因为下面打印了:

    ZeroDivisionError: integer division or modulo by zero
    

    根据错误类型ZeroDivisionError,我们判断,int(s)本身并没有出错,但是int(s)返回0,在计算10 / 0时出错,至此,找到错误源头。

    六、记录异常

    如果不捕获错误,自然可以让Python解释器来打印出错误堆栈,但程序也被结束了。既然我们能捕获错误,就可以把错误堆栈打印出来,然后分析错误原因,同时,让程序继续执行下去。

    Python内置的logging模块可以非常容易地记录错误信息:

    1. # err_logging.py
    2. import logging
    3. def foo(s):
    4. return 10 / int(s)
    5. def bar(s):
    6. return foo(s) * 2
    7. def main():
    8. try:
    9. bar('0')
    10. except Exception as e:
    11. logging.exception(e)
    12. main()
    13. print('END')

    同样是出错,但程序打印完错误信息后会继续执行,并正常退出:

    1. $ python3 err_logging.py
    2. ERROR:root:division by zero
    3. Traceback (most recent call last):
    4. File "err_logging.py", line 13, in main
    5. bar('0')
    6. File "err_logging.py", line 9, in bar
    7. return foo(s) * 2
    8. File "err_logging.py", line 6, in foo
    9. return 10 / int(s)
    10. ZeroDivisionError: division by zero
    11. END

    通过配置,logging还可以把错误记录到日志文件里,方便事后排查。

    七、自定义异常&抛出异常

    因为错误是class,捕获一个错误就是捕获到该class的一个实例。因此,错误并不是凭空产生的,而是有意创建并抛出的。Python的内置函数会抛出很多类型的错误,我们自己编写的函数也可以抛出错误。

    1. class MyException(Exception):
    2. pass
    3. try:
    4. pass
    5. except MyException as e:
    6. print("MyException异常被触发了", e)
    7. except Exception as e:
    8. print("Exception", e)

    上述代码在except中定义了捕获MyException异常,但他永远不会被触发。因为默认的那些异常都有特定的触发条件,例如:索引不存在、键不存在会触发IndexError和KeyError异常。

    对于我们自定义的异常,如果想要触发,则需要使用:raise MyException()类实现。

    1. class MyException(Exception):
    2. pass
    3. try:
    4. # 。。。
    5. raise MyException()
    6. # 。。。
    7. except MyException as e:
    8. print("MyException异常被触发了", e)
    9. except Exception as e:
    10. print("Exception", e)

    自定义异常类的时候可以加上异常信息属性

    1. class MyException(Exception):
    2. def __init__(self, msg, *args, **kwargs):
    3. super().__init__(*args, **kwargs)
    4. self.msg = msg
    5. try:
    6. raise MyException("xxx失败了")
    7. except MyException as e:
    8. print("MyException异常被触发了", e.msg)
    9. except Exception as e:
    10. print("Exception", e)

    或者

    1. class MyException(Exception):
    2. title = "请求错误"
    3. try:
    4. raise MyException()
    5. except MyException as e:
    6. print("MyException异常被触发了", e.title)
    7. except Exception as e:
    8. print("Exception", e)

    使用案例1:你我合作协同开发,你调用我写的方法。

    我定义了一个函数

    1. class EmailValidError(Exception):
    2. title = "邮箱格式错误"
    3. class ContentRequiredError(Exception):
    4. title = "文本不能为空错误"
    5. def send_email(email,content):
    6. if not re.match("\w+@live.com",email):
    7. raise EmailValidError()
    8. if len(content) == 0 :
    9. raise ContentRequiredError()
    10. # 发送邮件代码...
    11. # ...

    你调用我写的函数

    1. def execute():
    2. # 其他代码
    3. # ...
    4. try:
    5. send_email(...)
    6. except EmailValidError as e:
    7. pass
    8. except ContentRequiredError as e:
    9. pass
    10. except Exception as e:
    11. print("发送失败")
    12. execute()
    13. # 提示:如果想要写的简单一点,其实只写一个Exception捕获错误就可以了。

    使用案例2:在框架内部已经定义好,遇到什么样的错误都会触发不同的异常。

    1. import requests
    2. from requests import exceptions
    3. while True:
    4. url = input("请输入要下载网页地址:")
    5. try:
    6. res = requests.get(url=url)
    7. print(res)
    8. except exceptions.MissingSchema as e:
    9. print("URL架构不存在")
    10. except exceptions.InvalidSchema as e:
    11. print("URL架构错误")
    12. except exceptions.InvalidURL as e:
    13. print("URL地址格式错误")
    14. except exceptions.ConnectionError as e:
    15. print("网络连接错误")
    16. except Exception as e:
    17. print("代码出现错误", e)
    18. # 提示:如果想要写的简单一点,其实只写一个Exception捕获错误就可以了。

    案例3:按照规定去触发指定的异常,每种异常都具备被特殊的含义。

    八、with的用法

    With语句是什么?

    有一些任务,可能事先需要设置,事后做清理工作。对于这种场景,Python的with语句提供了一种非常方便的处理方式。其中一个很好的例子是文件处理,你需要获取一个文件句柄,从文件中读取数据,然后关闭文件句柄。
    如果不用with语句,代码如下:

    1. file = open("/tmp/foo.txt")
    2. data = file.read()
    3. file.close()

    这里有两个问题。一是可能忘记关闭文件句柄;二是文件读取数据发生异常,没有进行任何处理。下面是处理异常的加强版本:

    1. file = open("/tmp/foo.txt")
    2. try:
    3. data = file.read()
    4. finally:
    5. file.close()
    这段代码运行良好,但是太冗长。这时候with便体现出了优势。 除了有更优雅的语法,with还可以很好的处理上下文环境产生的异常。下面是with版本的代码:
    1. with open("/tmp/foo.txt") as file:
    2. data = file.read()

    是不是很简单?
    但是如果对with工作原理不熟悉的通许可能会和刚才的我一样,不懂其中原理
    那么下面我们简单看一下with的工作原理

    with是如何工作的?
    基本思想是:with所求值的对象必须有一个enter()方法,一个exit()方法。

    紧跟with**后面的语句被求值后,返回对象的**__enter__()方法被调用,这个方法的返回值将被赋值给as后面的变量。当with后面的代码块全部被执行完之后,将调用前面返回对象的exit()方法。

    下面是一个例子
     

    1. ######################
    2. ########with()##########
    3. ######################
    4. class Sample:
    5. def __enter__(self):
    6. print("in __enter__")
    7. return "Foo"
    8. def __exit__(self, exc_type, exc_val, exc_tb):
    9. #exc_type: 错误的类型
    10. #exc_val: 错误类型对应的值
    11. #exc_tb: 代码中错误发生的位置
    12. print("in __exit__")
    13. def get_sample():
    14. return Sample()
    15. with get_sample() as sample:
    16. print("Sample: " ,sample)

    运行

    分析运行过程:

    1. 进入这段程序,首先创建Sample类,完成它的两个成员函数enter ()、exit()的定义,然后顺序向下定义get_sample()函数.
    2. 进入with语句,调用get_sample()函数,返回一个Sample()类的对象,此时就需要进入Sample()类中,可以看到
    1. 1. __enter__()方法先被执行
    2. 2. __enter__()方法返回的值 - 这个例子中是"Foo",赋值给变量'sample'
    3. 3. 执行with中的代码块,打印变量"sample",其值当前为 "Foo"
    4. 4. 最后__exit__()方法被调用

    参考python的with用法 | kissdata

  • 相关阅读:
    Anaconda安装和配置
    断言与参数化
    牛客网《剑指offer》专栏刷题练习|锻炼递归思想|练习栈的使用
    【软件设计师-中级——刷题记录6(纯干货)】
    计算机毕业设计Java-酒店管理系统-(源码+系统+mysql数据库+lw文档)
    ES-OAS-ERP-电子政务-企业信息化
    vue2技能树(12)-路由守卫、动态路由、状态管理
    threejs(2)-Geometry进阶详解
    服务器内存总量和内存条有差异是什么问题 103.239.244.X
    【MySQL】-增删查改
  • 原文地址:https://blog.csdn.net/qq_34491508/article/details/133939656