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逐行读取文件内容的三种方法
Jan 20 Python
python实现socket客户端和服务端简单示例
Feb 24 Python
Python中DJANGO简单测试实例
May 11 Python
python基于xmlrpc实现二进制文件传输的方法
Jun 02 Python
详解Python中heapq模块的用法
Jun 28 Python
Python制作豆瓣图片的爬虫
Dec 28 Python
Python基于递归算法实现的汉诺塔与Fibonacci数列示例
Apr 18 Python
Python 变量类型详解
Oct 10 Python
使用python PIL库实现简单验证码的去噪方法步骤
May 10 Python
Python简易计算器制作方法代码详解
Oct 31 Python
基于python监控程序是否关闭
Jan 14 Python
Python中用pyinstaller打包时的图标问题及解决方法
Feb 17 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脚本数据库功能详解(下)
2006/10/09 PHP
phplock(php进程锁) v1.0 beta1
2009/11/24 PHP
php中http与https跨域共享session的解决方法
2014/12/20 PHP
Laravel源码解析之路由的使用和示例详解
2018/09/27 PHP
js控制CSS样式属性语法对照表
2012/12/11 Javascript
js获取当前页面路径示例讲解
2014/01/08 Javascript
基于jQuery实现表单提交验证
2014/11/24 Javascript
基于javascript实现的搜索时自动提示功能
2014/12/26 Javascript
浅谈js中变量初始化
2015/02/03 Javascript
最简单纯JavaScript实现Tab标签页切换的方式(推荐)
2016/07/25 Javascript
JS自定义函数实现时间戳转换成date的方法示例
2017/08/27 Javascript
详解Vue如何支持JSX语法
2017/11/10 Javascript
vue.js打包之后可能会遇到的坑!
2018/06/03 Javascript
Nodejs异步回调之异常处理实例分析
2018/06/22 NodeJs
laravel实现中文和英语互相切换的例子
2019/09/30 Javascript
vue transition 在子组件中失效的解决
2019/11/12 Javascript
JavaScript观察者模式原理与用法实例详解
2020/03/10 Javascript
js实现飞机大战小游戏
2020/08/26 Javascript
利用Python的Twisted框架实现webshell密码扫描器的教程
2015/04/16 Python
用Python将IP地址在整型和字符串之间轻松转换
2017/03/22 Python
Python爬虫_城市公交、地铁站点和线路数据采集实例
2018/01/10 Python
解决Python的str强转int时遇到的问题
2018/04/09 Python
Python简单计算文件MD5值的方法示例
2018/04/11 Python
python3学生名片管理v2.0版
2018/11/29 Python
Python常用的json标准库
2019/02/19 Python
python实现批量注册网站用户的示例
2019/02/22 Python
Python调用scp向服务器上传文件示例
2019/12/22 Python
python 两种方法修改文件的创建时间、修改时间、访问时间
2020/09/26 Python
HTML5 FileReader对象的具体使用方法
2020/05/22 HTML / CSS
苏格兰销售女装、男装和童装的连锁店:M&Co
2018/03/16 全球购物
毕业自我鉴定书
2014/03/24 职场文书
班委竞选演讲稿
2014/04/28 职场文书
家庭教育的心得体会
2014/09/01 职场文书
教师节学生演讲稿
2014/09/03 职场文书
搬迁通知
2015/04/20 职场文书
PHP遍历数组的6种方式总结
2021/11/17 PHP