python错误调试及单元文档测试过程解析


Posted in Python onDecember 19, 2019

这篇文章主要介绍了python错误调试及单元文档测试过程解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下

错误分为程序的错误和由用户错误的输入引起的错误,此外还有因为各种各样意外的情况导致的错误,比如在磁盘满的时候写入、从网络爬取东西的时候,网络断了。这类错误称为异常

错误处理

普通的错误处理机制就是在出错的时候返回一个错误代码,但是这样十分不方便,一是因为错误码是和正常结果一样的方式返回的,判断起来十分不方便,二是错误还需要一级一级的向上报,直到错误处理程序。

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

try:
  A#如果A中的代码执行过程中出错,就会执行B中的代码
except ZeroDivisionError as e:
  B
finally:
  C#C中的代码无论是否出错都会正常执行(可以不要这个)<br>。。。

如果错误有不同的类型,可以说使用多个except语句,每个语句处理一个类型的错误

另外,可以在except后面加一个else,如果没有出错,会执行else

Python 的错误其实也是一个类,所有的异常类型都是从BaseException类派生的

except在捕获错误时,不但捕获该类型的错误,而且还会把子类一网打尽

try:
  foo()
except ValueError as e:
  print('ValueError')
except UnicodeError as e:
  print('UnicodeError')
#第二个except永远也捕获不到UnicodeError,因为UnicodeError是ValueError的子类,如果有,也被第一个except给捕获了。

使用try...except还有一个巨大的好处,就是可以跨越多层调用,比如函数main()调用foo(),foo()调用bar(),结果bar()出错了,这时,只要main()捕获到了,就可以处理。也就是说,不需要在每个可能出错的地方去捕获错误,只要在合适的层次去捕获错误就可以了。这样一来,就大大减少了写try...except...finally的麻烦。

记录错误

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

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

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

抛出错误

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

如果要抛出错误,首先根据需要,可以定义一个错误的class,选择好继承关系,然后,用raise语句抛出一个错误的实例

只有在有必要的时候才定义我们自己的错误

另外一种错误处理

在try...excep捕获到异常后,还可以在except中使用 'raise‘把异常抛出去,以便于上级处理,如果raise语句不带参数,就会把异常原样抛出去,我们还可以通过raise 跟一个别的异常类型来将一种错误的类型转化为另外一种类型如:

try:
  10 / 0
except ZeroDivisionError:
  raise ValueError('input error!')

这种类型应该是一种合理的类型,而不应该将一种类型转化为另外一种不相干的类型

程序也可以主动抛出错误,让调用者来处理相应的错误。但是,应该在文档中写清楚可能会抛出哪些错误,以及错误产生的原因。

调试

断言

我们有事再调试的时候为了省事,就直接由print打印出变量的值,断言的作用和上面一样,凡是可以用print来辅助查看的地方,都可以用断言替代

断言可以加提示信息,

def foo(s):
  n = int(s)
  assert n != 0, 'n is zero!'#检查n是否是0,返回bool
  return 10 / n
 
def main():
  foo('0')

如果断言失败,assert语句本身就会抛出AssertionError:提示信息

启动Python解释器时可以用-O参数来关闭assert:

$ python -O err.py

使用pdb方式来调试

python -m pdb fortest.py#使用-m pdb 来启动调试
l #使用l来查看代码
n #使用n来执行一行代码
p 变量名#任何时候都可以输入p加变量名来查看变量
q#使用q退出

pdb.set_trace()

这个方法也是用pdb,但是不需要单步执行,我们只需要import pdb,然后,在可能出错的地方放一个pdb.set_trace(),就可以设置一个断点:

运行代码,程序会自动在pdb.set_trace()暂停并进入pdb调试环境,可以用命令p查看变量,或者用命令c继续运行:

IDE

虽然用IDE调试起来比较方便,但是最后你会发现,logging才是终极武器。

单元测试

为什么编写单元测试呢,因为在写好的程序可能在以后还需要修改,这时如果由单元测试,我们就能够保证修改后的程序在功能上和以前的相同,这一定程度上也减少了测试的繁杂性

这种以测试为驱动的开发模式最大的好处就是确保一个程序模块的行为符合我们设计的测试用例。在将来修改的时候,可以极大程度地保证该模块行为仍然是正确的。

接下来,作者举了一个例子来介绍了单元测试的编写模式,并且介绍了一些用到的函数

我们需要引入Python自带的测试模块unittest模块

import unittest

编写单元测试的时候,需要编写一个测试类,这个类从unittest.TestCase派生

def TestDict(unittest.TestCase):
  def test_init(self):
    pass

以test开头的方法就是测试方法,不以test开头的方法就不被认为是测试方法,运行单元测试的时候不会被执行

对每一类测试都需要编写一个测试方法,由于unittest.TestCase内置了很多判断,我们只需要断言这些输出是否是我们所需要的,最常用的断言就是assertEqual(),

self.assertEqual(abs(-1), 1) # 断言函数返回的结果与1相等
另一种重要的断言就是期待抛出指定类型的Error,比如通过d['empty']访问不存在的key时,断言会抛出KeyError:

with self.assertRaises(KeyError):
  value = d['empty']

运行单元测试

两种方法,一种直接在模块中加入

if __name__ == '__main__':
unittest.main()

另一种方法是在命令行通过参数-m unittest直接运行单元测试

这是推荐的做法,因为这样可以一次批量运行很多单元测试,并且,有很多工具可以自动来运行这些单元测试。

setUp和tearDown

这两个函数可以写在测试类中,作用就是再每个测试方法被调用之前会执行setUp(),被调用之后会执行tearDown(),可以把一些准备工作、和善后工作放到这些函数中。

  • 单元测试可以有效地测试某个程序模块的行为,是未来重构代码的信心保证。
  • 单元测试的测试用例要覆盖常用的输入组合、边界条件和异常。
  • 单元测试代码要非常简单,如果测试代码太复杂,那么测试代码本身就可能有bug。
  • 单元测试通过了并不意味着程序就没有bug了,但是不通过程序肯定有bug。

文档测试

文档测试就是运行写在注释中的实例代码

文档测试不能再调试(Debugger)模式下运行,否则会报错

PYDEV DEBUGGER WARNING:
sys.settrace() should not be used when the debugger is being used.
This may cause the debugger to stop working correctly.
If this is needed, please check:
http://pydev.blogspot.com/2007/06/why-cant-pydev-debugger-work-with.html
to see how to restore the debug tracing back correctly.
Call Location:
 File "c:\users\administrator.sc-201605202132\appdata\local\programs\python\python36\Lib\doctest.py", line 1480, in run
  sys.settrace(save_trace)

很多文档都有示例代码,可以把这些示例代码在Python的交互环境下运行。这些代码与其他说明可以写在注释中,然后,由一些工具来自动生成文档

def abs(n):
  '''
  Function to get absolute value of number.
   
  Example:
   
  >>> abs(1)
  1
  >>> abs(-1)
  1
  >>> abs(0)
  0
  '''
  return n if n >= 0 else (-n)

无疑更明确地告诉函数的调用者该函数的期望输入和输出。并且,Python内置的“文档测试”(doctest)模块可以直接提取注释中的代码并执行测试。

doctest严格按照Python交互式命令行的输入和输出来判断测试结果是否正确。只有测试异常的时候(即真正运行的结果和实例代码中的结果不一样的时候,就会报错),可以用...表示中间一大段烦人的输出。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Python 相关文章推荐
python实现归并排序算法
Nov 22 Python
python实现维吉尼亚算法
Mar 20 Python
使用python将mysql数据库的数据转换为json数据的方法
Jul 01 Python
Matplotlib绘制雷达图和三维图的示例代码
Jan 07 Python
mac在matplotlib中显示中文的操作方法
Mar 06 Python
Python面向对象程序设计之类和对象、实例变量、类变量用法分析
Mar 23 Python
Django-xadmin+rule对象级权限的实现方式
Mar 30 Python
三步解决python PermissionError: [WinError 5]拒绝访问的情况
Apr 22 Python
Python3爬虫里关于Splash负载均衡配置详解
Jul 10 Python
解决Windows下python和pip命令无法使用的问题
Aug 31 Python
scrapy redis配置文件setting参数详解
Nov 18 Python
python实现三次密码验证的示例
Apr 29 Python
Python3.5 win10环境下导入kera/tensorflow报错的解决方法
Dec 19 #Python
解决Python列表字符不区分大小写的问题
Dec 19 #Python
简单了解为什么python函数后有多个括号
Dec 19 #Python
解决Python使用列表副本的问题
Dec 19 #Python
python读写Excel表格的实例代码(简单实用)
Dec 19 #Python
python装饰器原理与用法深入详解
Dec 19 #Python
python列表生成器迭代器实例解析
Dec 19 #Python
You might like
分享五个PHP7性能优化提升技巧
2015/12/07 PHP
php加速缓存器opcache,apc,xcache,eAccelerator原理与配置方法实例分析
2020/03/02 PHP
javascript 触发HTML元素绑定的函数
2010/09/11 Javascript
收集的10个免费的jQuery相册
2011/02/26 Javascript
商城常用滚动的焦点图效果代码简单实用
2013/03/28 Javascript
JS验证邮箱格式是否正确的代码
2013/12/05 Javascript
php析构函数的具体用法小结
2014/03/11 Javascript
jquery attr方法获取input的checked属性问题
2014/05/26 Javascript
jquery实现个人中心导航菜单效果和美观都非常不错
2014/09/02 Javascript
CSS javascript 结合实现悬浮固定菜单效果
2015/08/23 Javascript
实例讲解js验证表单项是否为空的方法
2016/01/09 Javascript
全面解析Bootstrap布局组件应用
2016/02/22 Javascript
三种带箭头提示框总结实例
2016/06/14 Javascript
AngularJS中directive指令使用之事件绑定与指令交互用法示例
2016/11/22 Javascript
浅析使用BootStrap TreeView插件实现灵活配置快递模板
2016/11/28 Javascript
jQuery纵向导航菜单效果实现方法
2016/12/19 Javascript
详解react服务端渲染(同构)的方法
2017/09/21 Javascript
vue仿淘宝订单状态的tab切换效果
2020/06/23 Javascript
BootstrapValidator验证用户名已存在(ajax)
2019/11/08 Javascript
JavaScript canvas绘制圆弧与圆形
2020/02/18 Javascript
JavaScript实现鼠标经过表格某行时此行变色
2020/11/20 Javascript
[34:39]DOTA2上海特级锦标赛主赛事日 - 4 败者组第四轮#1COL VS EG第二局
2016/03/05 DOTA
Python中第三方库Requests库的高级用法详解
2017/03/12 Python
不到40行代码用Python实现一个简单的推荐系统
2019/05/10 Python
详解Python实现进度条的4种方式
2020/01/15 Python
Python对Tornado请求与响应的数据处理
2020/02/12 Python
在django admin详情表单显示中添加自定义控件的实现
2020/03/11 Python
python开发入门——列表生成式
2020/09/03 Python
英国泰坦旅游网站:全球陪同游览,邮轮和铁路旅行
2016/11/29 全球购物
美国按摩椅批发网站:Titan Chair
2018/12/27 全球购物
PHP面试题及答案一
2012/06/18 面试题
个人剖析材料及整改措施
2014/10/07 职场文书
神农溪导游词
2015/02/11 职场文书
退休教师欢送会致辞
2015/07/31 职场文书
Python Matplotlib绘制等高线图与渐变色扇形图
2022/04/14 Python
MySQL添加索引特点及优化问题
2022/07/23 MySQL