Python异常对代码运行性能的影响实例解析


Posted in Python onFebruary 08, 2018

前言

Python的异常处理能力非常强大,但是用不好也会带来负面的影响。我平时写程序的过程中也喜欢使用异常,虽然采取防御性的方式编码会更好,但是交给异常处理会起到偷懒作用。偶尔会想想异常处理会对性能造成多大的影响,于是今天就试着测试了一下。

Python异常(谷歌开源风格指南)

tip:

允许使用异常, 但必须小心。

定义:

异常是一种跳出代码块的正常控制流来处理错误或者其它异常条件的方式。

优点:

正常操作代码的控制流不会和错误处理代码混在一起. 当某种条件发生时, 它也允许控制流跳过多个框架. 例如, 一步跳出N个嵌套的函数, 而不必继续执行错误的代码。

缺点:

可能会导致让人困惑的控制流. 调用库时容易错过错误情况。

结论:

异常必须遵守特定条件:

像这样触发异常: raise MyException("Error message") 或者 raise MyException . 不要使用两个参数的形式( raise MyException, "Error message" )或者过时的字符串异常( raise "Error message" )。
模块或包应该定义自己的特定域的异常基类, 这个基类应该从内建的Exception类继承. 模块的异常基类应该叫做”Error”。

class Error(Exception):
  pass

永远不要使用 except: 语句来捕获所有异常, 也不要捕获 Exception 或者 StandardError , 除非你打算重新触发该异常, 或者你已经在当前线程的最外层(记得还是要打印一条错误消息). 在异常这方面, Python非常宽容, except: 真的会捕获包括Python语法错误在内的任何错误. 使用 except: 很容易隐藏真正的bug。

尽量减少try/except块中的代码量. try块的体积越大, 期望之外的异常就越容易被触发. 这种情况下, try/except块将隐藏真正的错误。

使用finally子句来执行那些无论try块中有没有异常都应该被执行的代码. 这对于清理资源常常很有用, 例如关闭文件。
当捕获异常时, 使用 as 而不要用逗号. 例如

try:
  raise Error
except Error as error:
  pass

设计实验方式

采取比较简单直观的对照实验。

先定义一个装饰器,用来计算每个函数执行所需时间:

def timer(func):
  import time
  def wrapper(*args, **kwargs):
    startTime = time.time()
    f = func(*args, **kwargs)
    endTime = time.time()
    passTime = endTime - startTime
    print "执行函数%s使用了%f秒" % (getattr(func, "__name__"), passTime)
    return f
  return wrapper

然后用该装饰器装饰测试的函数即可。

再定义一个叫do_something的函数,这个函数中就做一件事,把1赋值给变量a。在每个测试函数中,都会调用这个函数1000000次。

do_something:

def do_something():
  a = 1

我根据情况设计了不同的测试组:

测试组1(直接执行耗时操作):

@timer
def test1():
  for _ in xrange(1000000):
    do_something()

测试组2(耗时操作放在try中执行,不抛出错误):

@timer
def test2():
  try:
    for _ in xrange(1000000):
      do_something()
  except Exception:
    do_something()
  else:
    pass
  finally:
    pass

测试组3(try放耗时操作中,try每一次操作,不抛出错误):

@timer
def test3():
  for _ in xrange(1000000):
    try:
      do_something()
    except Exception:
      do_something()
    else:
      pass
    finally:
      pass

测试组4(try放耗时操作中,try每一次操作并进行异常处理(捕捉抛出的特定异常)):

@timer
def test4():
  zero = 0
  for _ in xrange(1000000):
    try:
      if zero == 0:
        raise ZeroDivisionError
    except ZeroDivisionError:
      do_something()
    else:
      pass
    finally:
      pass

测试组5(try放耗时操作中,try每一次操作并进行异常处理(捕捉所有异常 try…except BaseException)):

@timer
def test5():
  zero = 0
  for _ in xrange(1000000):
    try:
      if zero == 0:
        raise ZeroDivisionError
    except BaseException:
      do_something()
    else:
      pass
    finally:
      pass

测试组6(try放耗时操作中,try每一次操作并进行异常处理(捕捉所有异常 不带任何异常类型)):

@timer
def test6():
  zero = 0
  for _ in xrange(1000000):
    try:
      if zero == 0:
        raise ZeroDivisionError
    except:
      do_something()
    else:
      pass
    finally:
      pass

测试组7(耗时操作放在except中):

@timer
def test7():
  zero = 0
  try:
    if zero == 0:
      raise ZeroDivisionError
  except ZeroDivisionError:
    for _ in xrange(1000000):
      do_something()
  else:
    pass
  finally:
    pass

测试组8(防御式编码):

@timer
def test8():
  zero = 0
  for _ in xrange(1000000):
    if zero == 0:
      do_something()

执行结果

Python异常对代码运行性能的影响实例解析

对比结论

  • 通过对比1和2,可以得知直接执行耗时操作和耗时操作放在try中执行并无异常触发时性能消耗几乎是一样的。
  • 通过对比2和7,可以得知使用异常的使用无论是把代码放在 try 中执行还是在 except 中执行性能消耗几乎是一样的。
  • 通过对比2和3,可以得知当不抛出错误时,把try放耗时操作中比耗时操作放在try中性能消耗要略大。
  • 通过对比3和4,可以得知当使用try时无异常抛出跟使用try时抛出异常性能消耗几乎相差好几倍。
  • 通过对比4和5,可以得知try放耗时操作中时,try每一次操作并进行异常处理(捕捉抛出的特定异常)跟try每一次操作并进行异常处理(捕捉所有异常 try…except BaseException)性能消耗几乎是一样的。
  • 通过对比4和8,可以得知使用防御性方式编码比捕捉异常方式性能消耗几乎相差好几倍。
  • 通过对比5和6,可以得知捕捉所有异常(try…except)方式比捕捉所有异常(try…except BaseException)方式要略快。

总结

  1. 由以上对比结论,可以总结为:
  2. 无论是把代码放在 try 中执行还是在 except 中执行性能消耗几乎是一样的。
  3. 直接执行代码与放在try中执行且不抛出异常时性能消耗几乎是一样的,当然理论上try会消耗一点性能,可以忽略不计。
  4. 虽然try…except的方式比try…except BaseException和捕捉抛出的特定异常的方式要略快,但扔不建议采取这种方式,因为前者很容易隐藏真正的bug,从而带来严重后果。
  5. 通常要采取捕捉抛出的特定异常而不是捕捉所有异常,虽然二者性能消耗几乎一样。
  6. 防御性方式编码比捕捉异常方式性能消耗几乎相差好几倍,应尽量采取这种编程方式,提升性能并且更靠谱。

以上就是本文关于Python异常对代码运行性能的影响实例解析的全部内容,希望对大家有所帮助。感兴趣的朋友可以继续参阅本站其他相关专题,如有不足之处,欢迎留言指出。感谢朋友们对本站的支持!

Python 相关文章推荐
Python实现对百度云的文件上传(实例讲解)
Oct 21 Python
python format 格式化输出方法
Jul 16 Python
python 2.7 检测一个网页是否能正常访问的方法
Dec 26 Python
深入理解Python异常处理的哲学
Feb 01 Python
Python开发网站目录扫描器的实现
Feb 21 Python
Python3+OpenCV2实现图像的几何变换(平移、镜像、缩放、旋转、仿射)
May 13 Python
搭建python django虚拟环境完整步骤详解
Jul 08 Python
详解将Python程序(.py)转换为Windows可执行文件(.exe)
Jul 19 Python
快速查找Python安装路径方法
Feb 06 Python
解决jupyter notebook import error但是命令提示符import正常的问题
Apr 15 Python
python3字符串输出常见面试题总结
Dec 01 Python
python 字典和列表嵌套用法详解
Jun 29 Python
Python科学计算包numpy用法实例详解
Feb 08 #Python
Python多进程并发与多线程并发编程实例总结
Feb 08 #Python
Python的CGIHTTPServer交互实现详解
Feb 08 #Python
Python获取CPU、内存使用率以及网络使用状态代码
Feb 08 #Python
python实现二叉查找树实例代码
Feb 08 #Python
单链表反转python实现代码示例
Feb 08 #Python
Python测试人员需要掌握的知识
Feb 08 #Python
You might like
PHP模板引擎Smarty自定义变量调解器用法
2016/04/11 PHP
PhpStorm terminal无法输入命令的解决方法
2016/10/09 PHP
JavaScript网页制作特殊效果用随机数
2007/05/22 Javascript
使用Jquery打造最佳用户体验的登录页面的实现代码
2011/07/08 Javascript
Js参数值中含有单引号或双引号问题的解决方法
2013/11/06 Javascript
抛弃Nginx使用nodejs做反向代理服务器
2014/07/17 NodeJs
Javascript实现Array和String互转换的方法
2015/12/21 Javascript
JS组件Bootstrap dropdown组件扩展hover事件
2016/04/17 Javascript
js简单实现图片延迟加载的方法
2016/07/19 Javascript
JS实现鼠标滑过显示边框的菜单效果
2016/09/21 Javascript
ReactJs设置css样式的方法
2017/06/08 Javascript
vue-loader教程介绍
2017/06/14 Javascript
JavaScript+CSS相册特效实例代码
2017/09/07 Javascript
使用mint-ui开发项目的一些心得(分享)
2017/09/07 Javascript
vue树形结构获取键值的方法示例
2018/06/21 Javascript
react在安卓中输入框被手机键盘遮挡问题的解决方法
2018/09/03 Javascript
利用hasOwnProperty给数组去重的面试题分享
2018/11/05 Javascript
javascript实现计算指定范围内的质数示例
2018/12/29 Javascript
Vue源码解析之数据响应系统的使用
2019/04/24 Javascript
jquery 键盘事件 keypress() keydown() keyup()用法总结
2019/10/23 jQuery
[01:26]DOTA2荣耀之路2:iG,China
2018/05/24 DOTA
Python 元类使用说明
2009/12/18 Python
python中__call__方法示例分析
2014/10/11 Python
Django中间件实现拦截器的方法
2018/06/01 Python
Django中使用Celery的方法示例
2018/11/29 Python
python 内置函数汇总详解
2019/09/16 Python
解决Tensorflow 使用时cpu编译不支持警告的问题
2020/02/03 Python
python 3.8.3 安装配置图文教程
2020/05/21 Python
python 实现数据库中数据添加、查询与更新的示例代码
2020/12/07 Python
浅析Python打包时包含静态文件处理方法
2021/01/15 Python
苹果香港官方商城:Apple香港
2016/09/14 全球购物
作文批改评语大全
2014/04/23 职场文书
新学期红领巾广播稿
2014/10/04 职场文书
2014年卫生保健工作总结
2014/12/08 职场文书
2015年度质量工作总结报告
2015/04/27 职场文书
深入浅析python3 依赖倒置原则(示例代码)
2021/07/09 Python