博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
python中那个断言assert的优化
阅读量:4966 次
发布时间:2019-06-12

本文共 3919 字,大约阅读时间需要 13 分钟。

Python Assert 为何不尽如人意

Python中的断言用起来非常简单,你可以在assert后面跟上任意判断条件,如果断言失败则会抛出异常。

Copy
>>> assert 1 + 1 == 2 >>> assert isinstance('Hello', str) >>> assert isinstance('Hello', int) Traceback (most recent call last): File "", line 1, in 
AssertionError

其实assert看上去不错,然而用起来并不爽。就比如有人告诉你程序错了,但是不告诉哪里错了。很多时候这样的assert还不如不写,写了我就想骂娘。直接抛一个异常来得更痛快一些。

改进方案 #1

一个稍微改进一丢丢的方案就是把必要的信息也放到assert语句后面,比如这样。

Copy
>>> s = "nothin is impossible.">>> key = "nothing" >>> assert key in s, "Key: '{}' is not in Target: '{}'".format(key, s) Traceback (most recent call last): File "", line 1, in 
AssertionError: Key: 'nothing' is not in Target: 'nothin is impossible.'

看上去还行吧,但是其实写的很蛋疼。假如你是一名测试汪,有成千上万的测试案例需要做断言做验证,相信你面对以上做法,心中一定有千万只那种马奔腾而过。

改进方案 #2

不管你是你是搞测试还是开发的,想必听过不少测试框架。你猜到我要说什么了吧?对,不用测试框架里的断言机制,你是不是洒。

py.test

 是一个轻量级的测试框架,所以它压根就没写自己的断言系统,但是它对Python自带的断言做了强化处理,如果断言失败,那么框架本身会尽可能多地提供断言失败的原因。那么也就意味着,用py.test实现测试,你一行代码都不用改。

Copy
import pytestdef test_case(): expected = "Hello" actual = "hello" assert expected == actual if __name__ == '__main__': pytest.main() """ ================================== FAILURES =================================== __________________________________ test_case __________________________________ def test_case(): expected = "Hello" actual = "hello" > assert expected == actual E assert 'Hello' == 'hello' E - Hello E ? ^ E + hello E ? ^ assertion_in_python.py:7: AssertionError ========================== 1 failed in 0.05 seconds =========================== """"

unittest

Python自带的单元测试框架就有了自己的断言方法self.assertXXX(),而且不推荐使用assert XXX语句。

Copy
import unittestclass TestStringMethods(unittest.TestCase): def test_upper(self): self.assertEqual('foo'.upper(), 'FoO') if __name__ == '__main__': unittest.main() """ Failure Expected :'FOO' Actual :'FoO' Traceback (most recent call last): File "assertion_in_python.py", line 6, in test_upper self.assertEqual('foo'.upper(), 'FoO') AssertionError: 'FOO' != 'FoO' """

ptest

我非常喜欢,感谢Karl大神写了这么一个测试框架。ptest中的断言可读性很好,而且通过IDE的智能提示你能轻松完成各种断言语句。

Copy
from ptest.decorator import *from ptest.assertion import * @TestClass() class TestCases: @Test() def test1(self): actual = 'foo' expected = 'bar' assert_that(expected).is_equal_to(actual) """ Start to run following 1 tests: ------------------------------ ... [demo.assertion_in_python.TestCases.test1@Test] Failed with following message: ... AssertionError: Unexpectedly that the str 
is not equal to str
. """

改进方案 #3

不仅仅是你和我对Python中的断言表示不满足,所以大家都争相发明自己的assert包。在这里我强烈推荐 这个包,它异常强大而且好评如潮。

Copy
pip install assertpy

看例子:

Copy
from assertpy import assert_thatdef test_something(): assert_that(1 + 2).is_equal_to(3) assert_that('foobar')\ .is_length(6)\ .starts_with('foo')\ .ends_with('bar') assert_that(['a', 'b', 'c'])\ .contains('a')\ .does_not_contain('x')

从它的 文档上你会发现它支持了几乎你能想到的所有测试场景,包括但不限于以下列表。

  • Strings
  • Numbers
  • Lists
  • Tuples
  • Dicts
  • Sets
  • Booleans
  • Dates
  • Files
  • Objects

而且它的断言信息简洁明了,不多不少。

Copy
Expected 
to be of length <4>, but was <3>.Expected
to be empty string, but was not. Expected
, but was not. Expected
to contain only digits, but did not. Expected <123> to contain only alphabetic chars, but did not. Expected
to contain only uppercase chars, but did not. Expected
to contain only lowercase chars, but did not. Expected
to be equal to
, but was not. Expected
to be not equal to
, but was. Expected
to be case-insensitive equal to
, but was not.

在发现assertpy之前我也想写一个类似的包,尽可能通用一些。但是现在,我为毛要重新去造轮子?完全没必要!

总结

断言在软件系统中有非常重要的作用,写的好可以让你的系统更稳定,也可以让你有更多面对(女)对象的时间,而不是在调试代码。

Python中默认的断言语句其实还有一个作用,如果你写了一个类型相关的断言,IDE会把这个对象当成这种类型,这时候智能提示就有如神助。

要不要把内置的断言语句换成可读性更好功能更强大的第三方断言,完全取决于实际情况。比如你真的需要验证某个东西并且很关心验证结果,那么必须不能用简单的assert;如果你只是担心某个点可能有坑或者让IDE认识某个对象,用内置的assert既简单又方便。

所以说,项目经验还是蛮重要的。

作者:

出处:

本站使用「」创作共享协议,转载请在文章明显位置注明作者及出处。

转载于:https://www.cnblogs.com/hls91/p/10768776.html

你可能感兴趣的文章
springMVC之mvc:interceptors拦截器的用法
查看>>
Java之美[从菜鸟到高手演变]系列之博文阅读导航
查看>>
C#将科大讯飞语音合成文件转换为MULAW音频格式
查看>>
解决使用Rails中使用中文时出现"Your template was not saved as valid UTF-8"的问题
查看>>
LoadRunner配置方案
查看>>
loadrunner解决“服务器正在运行中”方法
查看>>
界面编程之QT的线程20180731
查看>>
三元环计数问题
查看>>
页面内嵌iframe 防止session过期
查看>>
c#冒泡算法
查看>>
socketv 验证客户端链接的合法性
查看>>
iis 启用父目录路径访问
查看>>
javaweb学习总结(四十三)——Filter高级开发
查看>>
paip.函数方法回调机制跟java php python c++的实现
查看>>
Atitit 自然语言处理原理与实现 attilax总结
查看>>
数据库——JDBC
查看>>
第22章:MongoDB-聚合操作--聚合管道--$out
查看>>
Python 列表
查看>>
SpringMVC转发和重定向区别!
查看>>
io.js的六大新特性
查看>>