第28章 Python3 错误和异常教程

作为 Python 初学者,在刚学习 Python 编程时,经常会看到一些报错信息,在前面咱们没有提及,这章节咱们会专门介绍。

Python 有两种错误很容易辨认:语法错误和异常。

Python assert(断言)用于判断一个表达式,在表达式条件为 false 的时候触发异常。

语法错误

Python 的语法错误或者称之为解析错,是初学者经常碰到的,如下实例


    >>> while True print('Hello world')
    
      File "<stdin>", line 1, in ?
    
        while True print('Hello world')
    
                       ^
    
    SyntaxError: invalid syntax

这个例子中,函数 print() 被检查到有错误,是它前面缺少了一个冒号 : 。

语法分析器指出了出错的一行,并且在最先找到的错误的位置标记了一个小小的箭头。

异常

即便 Python 程序的语法是正确的,在运行它的时候,也有可能发生错误。运行期检测到的错误被称为异常。

大多数的异常都不会被程序处理,都以错误信息的形式展现在这里:


    实例 
     
    
    >>> 10 * (1/0)             # 0 不能作为除数,触发异常
    
    Traceback (most recent call last):
    
      File "<stdin>", line 1, in ?
    
    ZeroDivisionError: division by zero
    
    >>> 4 + spam*3             # spam 未定义,触发异常
    
    Traceback (most recent call last):
    
      File "<stdin>", line 1, in ?
    
    NameError: name 'spam' is not defined
    
    >>> '2' + 2               # int 不能与 str 相加,触发异常
    
    Traceback (most recent call last):
    
      File "<stdin>", line 1, in <module>
    
    TypeError: can only concatenate str (not "int") to str

异常以不同的类型出现,这些类型都作为信息的一部分打印出来: 例子中的类型有 ZeroDivisionError,NameError 和 TypeError。

错误信息的前面部分显示了异常发生的上下文,并以调用栈的形式显示具体信息。

异常处理

try/except

异常捕捉可以使用 try/except 语句。

以下例子中,让用户输入一个合法的整数,但是允许用户中断这个程序(使用 Control-C 或者操作系统提供的方法)。用户中断的信息会引发一个 KeyboardInterrupt 异常。


    while True:
    
        try:
    
            x = int(input("请输入一个数字: "))
    
            break
    
        except ValueError:
    
            print("您输入的不是数字,请再次尝试输入!")

try 语句按照如下方式工作;

  • 首先,执行 try 子句(在关键字 try 和关键字 except 之间的语句)。
  • 如果没有异常发生,忽略 except 子句,try 子句执行后结束。
  • 如果在执行 try 子句的过程中发生了异常,那么 try 子句余下的部分将被忽略。如果异常的类型和 except 之后的名称相符,那么对应的 except 子句将被执行。
  • 如果一个异常没有与任何的 except 匹配,那么这个异常将会传递给上层的 try 中。

一个 try 语句可能包含多个except子句,分别来处理不同的特定的异常。最多只有一个分支会被执行。

处理程序将只针对对应的 try 子句中的异常进行处理,而不是其他的 try 的处理程序中的异常。

一个except子句可以同时处理多个异常,这些异常将被放在一个括号里成为一个元组,例如:


    except (RuntimeError, TypeError, NameError):
    
        pass

最后一个except子句可以忽略异常的名称,它将被当作通配符使用。阁下您可以使用这种方法打印一个错误信息,然后再次把异常抛出。


    import sys
    
    try:
    
        f = open('myfile.txt')
    
        s = f.readline()
    
        i = int(s.strip())
    
    except OSError as err:
    
        print("OS error: {0}".format(err))
    
    except ValueError:
    
        print("Could not convert data to an integer.")
    
    except:
    
        print("Unexpected error:", sys.exc_info()[0])
    
        raise

try/except…else

try/except 语句还有一个非必输的 else 子句,如果使用这个子句,那么必须放在所有的 except 子句之后。

else 子句将在 try 子句没有发生任何异常的时候执行。

以下实例在 try 语句中判断文件是否可以打开,如果打开文件时正常的没有发生异常则执行 else 部分的语句,读取文件内容:


    for arg in sys.argv[1:]:
    
        try:
    
            f = open(arg, 'r')
    
        except IOError:
    
            print('cannot open', arg)
    
        else:
    
            print(arg, 'has', len(f.readlines()), 'lines')
    
            f.close()

使用 else 子句比把所有的语句都放在 try 子句里面要好,这样可以避免一些意想不到,而 except 又无法捕获的异常。

异常处理并不仅仅处理那些直接发生在 try 子句中的异常,而且还能处理子句中调用的函数(甚至间接调用的函数)里抛出的异常。例如:


    >>> def this_fails():
    
            x = 1/0
    
    >>> try:
    
            this_fails()
    
        except ZeroDivisionError as err:
    
            print('Handling run-time error:', err)
    
       
    
    Handling run-time error: int division or modulo by zero

try-finally 语句

try-finally 语句无论是否发生异常都将执行最后的代码。

以下实例中 finally 语句无论异常是否发生都会执行:


    实例 
    try:
    
        phlcsdn2023()
    
    except AssertionError as error:
    
        print(error)
    
    else:
    
        try:
    
            with open('file.log') as file:
    
                read_data = file.read()
    
        except FileNotFoundError as fnf_error:
    
            print(fnf_error)
    
    finally:
    
        print('这句话,无论异常是否发生都会执行。')


抛出异常

Python 使用 raise 语句抛出一个指定的异常。

raise语法格式如下:


    raise [Exception [, args [, traceback]]]

以下实例如果 x 大于 5 就触发异常:


    x = 10
    
    if x > 5:
    
        raise Exception('x 不能大于 5。x 的值为: {}'.format(x))

执行以上代码会触发异常:


    Traceback (most recent call last):
      File "test.py", line 3, in <module>
        raise Exception('x 不能大于 5。x 的值为: {}'.format(x))
    Exception: x 不能大于 5。x 的值为: 10

raise 唯一的一个参数指定了要被抛出的异常。它必须是一个异常的实例或者是异常的类(也就是 Exception 的子类)。

如果阁下您只想知道这是否抛出了一个异常,并不想去处理它,那么一个简单的 raise 语句就可以再次把它抛出。


    >>> try:
    
            raise NameError('HiThere')
    
        except NameError:
    
            print('An exception flew by!')
    
            raise
    
       
    
    An exception flew by!
    
    Traceback (most recent call last):
    
      File "<stdin>", line 2, in ?
    
    NameError: HiThere


用户自定义异常

阁下您可以通过创建一个新的异常类来拥有自己的异常。异常类继承自 Exception 类,可以直接继承,或者间接继承,例如:


    >>> class MyError(Exception):
    
            def __init__(self, value):
    
                self.value = value
    
            def __str__(self):
    
                return repr(self.value)
    
    >>> try:
    
            raise MyError(2*2)
    
        except MyError as e:
    
            print('My exception occurred, value:', e.value)
    
       
    
    My exception occurred, value: 4
    
    >>> raise MyError('oops!')
    
    Traceback (most recent call last):
    
      File "<stdin>", line 1, in ?
    
    __main__.MyError: 'oops!'

在这个例子中,类 Exception 默认的 __init__() 被覆盖。

当创建一个模块有可能抛出多种不同的异常时,一种通常的做法是为这个包建立一个基础异常类,然后基于这个基础类为不同的错误情况创建不同的子类:


    class Error(Exception):
    
        """Base class for exceptions in this module."""
    
        pass
    
    class InputError(Error):
    
        """Exception raised for errors in the input.     Attributes:         expression -- input expression in which the error occurred         message -- explanation of the error     """
    
        def __init__(self, expression, message):
    
            self.expression = expression
    
            self.message = message
    
    
    
    class TransitionError(Error):
    
        """Raised when an operation attempts a state transition that's not     allowed.     Attributes:         previous -- state at beginning of transition         next -- attempted new state         message -- explanation of why the specific transition is not allowed     """
    
        def __init__(self, previous, next, message):
    
            self.previous = previous
    
            self.next = next
    
            self.message = message

大多数的异常的名字都以"Error"结尾,就跟标准的异常命名一样。


定义清理行为

try 语句还有另外一个非必输的子句,它定义了无论在任何情况下都会执行的清理行为。 例如:


    >>> try:
    
    ...     raise KeyboardInterrupt
    
    ... finally:
    
    ...     print('Goodbye, world!')
    
    ... 
    
    Goodbye, world!
    
    Traceback (most recent call last):
    
      File "<stdin>", line 2, in <module>
    
    KeyboardInterrupt

以上例子不管 try 子句里面有没有发生异常,finally 子句都会执行。

如果一个异常在 try 子句里(或者在 except 和 else 子句里)被抛出,而又没有任何的 except 把它截住,那么这个异常会在 finally 子句执行后被抛出。

下面是一个更加复杂的例子(在同一个 try 语句里包含 except 和 finally 子句):


    >>> def divide(x, y):
    
            try:
    
                result = x / y
    
            except ZeroDivisionError:
    
                print("division by zero!")
    
            else:
    
                print("result is", result)
    
            finally:
    
                print("executing finally clause")
    
       
    
    >>> divide(2, 1)
    
    result is 2.0
    
    executing finally clause
    
    >>> divide(2, 0)
    
    division by zero!
    
    executing finally clause
    
    >>> divide("2", "1")
    
    executing finally clause
    
    Traceback (most recent call last):
    
      File "<stdin>", line 1, in ?
    
      File "<stdin>", line 3, in divide
    
    TypeError: unsupported operand type(s) for /: 'str' and 'str'


预定义的清理行为

一些对象定义了标准的清理行为,无论系统是否成功的使用了它,一旦不需要它了,那么这个标准的清理行为就会执行。

下面这个例子展示了尝试打开一个文件,然后把内容打印到屏幕上:


    for line in open("myfile.txt"):
    
        print(line, end="")

以上这段代码的问题是,当执行完毕后,文件会保持打开状态,并没有被关闭。

关键词 with 语句就可以保证诸如文件之类的对象在使用完之后一定会正确的执行他的清理方法:


    with open("myfile.txt") as f:
    
        for line in f:
    
            print(line, end="")

以上这段代码执行完毕后,就算在处理过程中出问题了,文件 f 总是会关闭。

本专栏所有文章

第1章 Python3 教程第2章 Python3 简介教程
第3章 Python3 环境搭建教程第4章 Python3 VScode教程
第5章 Python3 基础语法教程第6章 Python3 基本数据类型教程
第7章 Python3 数据类型转换教程第8章 Python3 推导式教程
第9章 Python3 解释器教程第10章 Python3 注释教程
第11章 Python3 运算符教程第12章 Python3 数字(Number)教程
第13章 Python3 字符串教程第14章 Python3 列表教程
第15章 Python3 元组教程第16章 Python3 字典教程
第17章 Python3 集合教程第18章 Python3 编程第一步教程
第19章 Python3 条件控制教程第20章 Python3 循环语句教程
第21章 Python3 迭代器与生成器教程第22章 Python3 函数教程
第23章 Python3 数据结构教程第24章 Python3 模块教程
第25章 Python3 输入和输出教程第26章 Python3 File教程
第27章 Python3 OS教程第28章 Python3 错误和异常教程
第29章 Python3 面向对象教程第30章 Python3 命名空间/作用域教程
第31章 Python3 标准库概览教程第32章 Python3 实例教程
第33章 Python 测验教程第34章 Python3 正则表达式教程
第35章 Python3 CGI编程教程第36章 Python3 MySQL(mysql-connector)教程
第37章 Python3 MySQL(PyMySQL)教程第38章 Python3 网络编程教程
第39章 Python3 SMTP发送邮件教程第40章 Python3 多线程教程
第41章 Python3 XML 解析教程第42章 Python3 JSON教程
第43章 Python3 日期和时间教程第44章 Python3 内置函数教程
第45章 Python3 MongoDB教程第46章 Python3 urllib教程
第47章 Python uWSGI 安装配置教程第48章 Python3 pip教程
第49章 Python3 operator教程第50章 Python math教程
第51章 Python requests教程第52章 Python random教程
第53章 Python3 os.replace() 方法教程

寄语

本文有 phlcsdn2023 原创,欢迎点赞、转载,博客地址:https://blog.csdn.net/phlcsdn2023

  • 我们经历挫折,是因为上天不允许我们停止努力面对强大的对手,明知不敌,也要毅然亮剑,即使倒下,也要化成一座山,一道岭。
  • 年少轻狂的我们,鲜衣怒马,却不知不觉奔向了时空布好的囚笼。当光阴流过,记忆沉淀,回看那一段疯狂岁月,只剩下心底隐隐作痛的伤,和不愿褪去却不再鲜明的淡淡执着。
  • 人不自主,必要人主,越是人主,越不自主。越接近人,越能自主,则无需人主。
  • 美是善良和诚挚之母。前苏联教育家苏霍姆林斯基
  • 人生就是一场没有尽头的修行,人生的大智慧就是修养出宽容来。时间没有尽头,生命有其长短。在无限的时光隧道里,计较太多就是对有限的生命的一种亵渎。因为计较,我们缺失了宽容,从而错过了幸福。人生都是为幸福而来,却有很多人含恨而终。修行,于苦难中回味甘甜,于泪光中瞥见最美的笑容。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
以下是100个Python小技巧: 1. 使用for循环可以遍历列表、字符串等可迭代对象。 2. 使用range()函数生成指定范围的数字序列。 3. 使用enumerate()函数可以同时获得元素索引和值。 4. 使用zip()函数可以同时遍历多个可迭代对象。 5. 使用列表推导式可以快速生成列表。 6. 使用集合可以进行高效的集合操作。 7. 使用字典可以进行键值对的映射。 8. 使用切片可以快速获取列表、字符串的子序列。 9. 使用函数可以封装可复用的代码块。 10. 使用类可以实现面向对象的编程。 11. 使用模块可以组织和管理代码。 12. 使用异常处理可以捕获和处理程序的错误。 13. 使用with语句可以自动管理资源的释放。 14. 使用装饰器可以增强函数的功能。 15. 使用生成器可以按需计算大量数据。 16. 使用协程可以实现异步编程。 17. 使用多线程可以并发执行任务。 18. 使用多进程可以充分利用多核处理器。 19. 使用文件操作可以读写文件。 20. 使用正则表达式可以进行复杂的文本匹配。 21. 使用time模块可以获取当前时间和进行时间操作。 22. 使用random模块可以生成随机数。 23. 使用math模块可以进行数学运算。 24. 使用json模块可以进行JSON数据的处理。 25. 使用pickle模块可以进行对象的序列化和反序列化。 26. 使用os模块可以进行文件和目录的操作。 27. 使用sys模块可以获取和修改Python解释器的运行时环境。 28. 使用re模块可以进行正则表达式匹配。 29. 使用argparse模块可以解析命令行参数。 30. 使用logging模块可以进行日志记录。 31. 使用unittest模块可以编写和执行单元测试。 32. 使用requests库可以发送HTTP请求。 33. 使用BeautifulSoup库可以解析HTML文档。 34. 使用numpy库可以进行数组和矩阵计算。 35. 使用pandas库可以进行数据处理和分析。 36. 使用matplotlib库可以进行数据可视化。 37. 使用scikit-learn库可以进行机器学习。 38. 使用tensorflow库可以进行深度学习。 39. 使用flask库可以构建Web应用。 40. 使用Django库可以构建全功能的Web应用。 41. 使用SQLite数据库可以进行轻量级的数据存储。 42. 使用MySQL数据库可以进行关系型数据存储。 43. 使用MongoDB数据库可以进行文档型数据存储。 44. 使用Redis数据库可以进行缓存和键值存储。 45. 使用Elasticsearch可以进行搜索和分析。 46. 使用OpenCV库可以进行图像处理和计算机视觉。 47. 使用pygame库可以进行游戏开发。 48. 使用tkinter库可以进行桌面应用程序开发。 49. 使用wxPython库可以进行跨平台的GUI开发。 50. 使用Flask-RESTful可以构建RESTful API。 51. 使用Celery可以进行异步任务的调度和执行。 52. 使用pytest可以进行更简洁和灵活的单元测试。 53. 使用Selenium可以进行Web自动化测试。 54. 使用Faker可以生成随机的测试数据。 55. 使用IPython可以进行交互式的开发和调试。 56. 使用Jupyter Notebook可以进行数据分析和可视化。 57. 使用Spyder可以进行科学计算和Python开发。 58. 使用cookiecutter可以快速构建项目模板。 59. 使用virtualenv可以创建和管理Python虚拟环境。 60. 使用pip可以安装和管理Python包。 61. 使用pyenv可以管理多个Python版本。 62. 使用conda可以创建和管理Python环境。 63. 使用autopep8可以自动格式化Python代码。 64. 使用black可以自动格式化Python代码。 65. 使用flake8可以检查Python代码是否符合PEP8规范。 66. 使用bandit可以检查Python代码中的安全漏洞。 67. 使用isort可以自动排序Python导入语句。 68. 使用mypy可以进行静态类型检查。 69. 使用pylint可以进行代码质量检查。 70. 使用pyinstaller可以将Python程序打包成可执行文件。 71. 使用cx_Freeze可以将Python程序打包成可执行文件。 72. 使用py2exe可以将Python程序打包成可执行文件。 73. 使用pyodbc可以连接和操作数据库。 74. 使用paramiko可以进行SSH远程操作。 75. 使用fabric可以进行任务的自动化部署。 76. 使用pytest-django可以简化Django的单元测试。 77. 使用django-rest-framework可以快速构建RESTful API。 78. 使用scrapy可以进行Web爬虫。 79. 使用sqlalchemy可以进行高级数据库操作。 80. 使用pymongo可以连接和操作MongoDB数据库。 81. 使用tqdm可以在循环中显示进度条。 82. 使用click可以构建命令行接口。 83. 使用cProfile可以进行性能分析。 84. 使用line_profiler可以逐行分析代码性能。 85. 使用memory_profiler可以分析内存使用情况。 86. 使用profilehooks可以进行分析函数调用性能。 87. 使用PyInstaller可以将Python程序打包成可执行文件。 88. 使用Openpyxl可以读写Excel文件。 89. 使用Pillow可以进行图像处理。 90. 使用pyautogui可以进行图像识别和自动化操作。 91. 使用pytesseract可以进行文字识别。 92. 使用pywin32可以操作Windows系统API。 93. 使用pyserial可以进行串口通信。 94. 使用opencv-python可以进行图像处理。 95. 使用catboost可以进行梯度提升决策树算法。 96. 使用fasttext可以进行文本分类和词向量训练。 97. 使用gensim可以进行文本相似度计算。 98. 使用jieba可以进行中文分词。 99. 使用lightgbm可以进行梯度提升算法。 100. 使用xgboost可以进行梯度提升算法。 这些小技巧涵盖了Python的各个方面,包括语法、标准库、第三方库和常见开发任务。通过学习和应用这些技巧,可以提高Python编程的效率和质量。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

你得不到的念想

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

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

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

打赏作者

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

抵扣说明:

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

余额充值