`

Stop Handling Exceptions: Your Code Will Run Faster

    博客分类:
  • Java
阅读更多

Recently there was a bigger discussion at  dynaTrace   around the cost of exceptions. When working with customers we very often find a lot of exceptions they are not aware of. After removing these exceptions, the code runs significantly faster than before. This creates the assumption that using exceptions in your code comes with a significant performance overhead. The implication would be that you better avoid using exceptions. As exceptions are an important construct for handling error situation, avoiding exceptions completely does not seem to be good solution. All in all this was reason enough to have a closer look at the costs of throwing exceptions.

The Experiment

I based my experiment on a simple piece of code that randomly throws an exception. This is not a really scientifically-profound measurement and we also don’t know what the HotSpot compiler does with the code as it runs. Nevertheless it should provide us with some basic insights.

public class ExceptionTest {
 
  public long maxLevel = 20;
 
  public static void main (String ... args){
 
    ExceptionTest test = new ExceptionTest();
 
    long start = System.currentTimeMillis();
    int count = 10000;
    for (int i= 0; i < count; i++){
      try { 
        test.doTest(2, 0);
      }catch (Exception ex){
//        ex.getStackTrace();
      }
    }
    long diff = System.currentTimeMillis() - start;
    System.out.println(String.format("Average time for invocation: %1$.5f",((double) diff)/count));
  }
 
  public void doTest (int i, int level){
      if (level < maxLevel){
        try {
          doTest (i, ++level);
        }
        catch (Exception ex){
//        ex.getStackTrace();
          throw new RuntimeException ("UUUPS", ex);
        }
      }
      else {
        if (i > 1) {
          throw new RuntimeException("Ups".substring(0, 3));
        }
      }
  }
}
 

The Result

The result was very interesting. The cost of throwing and catching an exception seems to be rather low. In my sample it was about 0.002ms per Exception. This can more or less be neglected unless you really throw too many exceptions – and too many means we are talking about 100.000 or more.

While these results show that exception handling itself is not affecting code performance, it leaves open the question: what is responsible for the huge performance impact of exceptions? So obviously I was missing something – something important.

After thinking about it again, I realized that I was missing an important part of exception handling. I missed out the part on what you do when exceptions occur. In most cases you – hopefully – do not just catch the exception and that’s it. Normally you try to compensate for the problem and keep the application functioning for your end users. So the point I was missing was the compensation code that is executed for handling an exception. Depending on what this code is doing the performance penalty can become quite significant. In some cases this might mean retrying to connect to a server in other cases it might mean using a default fallback solution that is providing a far less-performing solution.

While this seemed to be a good explanation for the behavior we saw in many scenarios, I thought I am not done yet with the analysis. I had the feeling that there is something else that I was missing here.

Stack Traces

Still curious about this problem I looked into how the situation changes when I collect stack traces. This is what very often happens. You log an exception and its stack trace to try to figure out what the problem is.

I therefore modified my code to now get the stack trace of an exception as well. This changed the situation dramatically. Getting the stack trace of an exception had a 10x higher impact on the performance than just catching and throwing them. So while stack traces help to understand where and possibly also why a problem occurred, they come with a performance penalty.

The impact here is often very high as we are not talking about a single stack trace. In most cases exceptions are thrown – and caught – at multiple levels. Let us look at a simple example of a Web Service client connecting to a server. First there is an exception at the Java library level for the failed connection. Then there is a framework exception for the failed client and then there might be an application-level exception that some business logic invocation failed. This now sums up to three stack traces being collected.

In most cases you should see them in your log files or application output. Writing these potentially long stack traces again comes with some performance impact. At least you normally see and you can react to them if you look at your log files regularly – which is something you do, don’t you?  ;-)

In some cases I have seen even worse behavior due to some incorrect logging code. Instead of checking whether a certain log level is enabled by calling log.isxxEnabled () first, developers just call logging methods. When this happens, logging code is always executed including getting stack traces of exceptions. As the log level however is set too low they never show up anywhere you might not even be aware of them. Checking for log levels first should be a general rule as it also avoids unnecessary object creation.

Conclusion

Not using exceptions because of their potential performance impact is a bad idea. Exceptions help to provide a uniform way to cope with runtime problems and they help to write clean code. You however need to trace the number of exceptions that are thrown in your code. Although they might be caught they can still have a significant performance impact. In dynaTrace we, by default, track thrown exceptions – and in many cases people are surprised by what is going on in their code and what the performance impact is in resolving them.

While exception usage is good you should avoid capturing too many stack traces. In many cases they are not even necessary to understand the problem – especially if they cover a problem you already expect. The exception message therefore might prove as being enough information. I get enough out of a Connection refused message so I do not need the full stack trace into the internal of the java.net call stack.

 

分享到:
评论

相关推荐

    Exceptions:探索异常

    在编程世界中,异常是程序运行时遇到的不正常情况,它们中断了正常的代码执行流程。在JavaScript中,异常处理是通过try...catch语句来实现的,这使得程序员能够优雅地处理错误,而不是让程序突然崩溃。...

    better-exceptions:Python中漂亮而有用的异常

    通过pip安装better_exceptions : $ pip install better_exceptions 并将BETTER_EXCEPTIONS环境变量设置为任何值: export BETTER_EXCEPTIONS=1 # Linux / OSX setx BETTER_EXCEPTIONS 1 # Windows 而已! ...

    Laravel-Exceptions:为Laravel提供强大的错误响应系统

    Laravel例外Laravel Exceptions由创建并维护,并为开发和生产提供了强大的错误响应系统。 它可以选择将软件包用于开发错误页面。 随时检查,,,,和。安装Laravel异常需要 7.2-8.0。 此特定版本支持Laravel 7-8。 ...

    Android代码-Sofa

    Handling the exceptions: try { something; } catch(Exception e) { StackOverflow.search(e); } Sofa So probably you're tired of copy pasting logcat exceptions and stack traces into search, right? Now ...

    PyPI 官网下载 | tidevice-0.4.8.tar.gz

    《PyPI官网下载:tidevice-0.4.8.tar.gz——探索Python在云原生环境中的应用》 PyPI(Python Package Index)是Python开发者的重要资源库,提供了丰富的Python库供全球开发者下载和使用。本文将围绕“tidevice-...

    django-http-exceptions:django的HTTP例外

    什么是django-http-exceptions ? 对于您的django视图,这是很可笑的例外。 到底有什么好处呢? 这使得 def some_function (): raise SomeError def view ( request ): try : response = some_function () ...

    vertx-https-exceptions:https的转载者

    【标题】"vertx-https-exceptions:https的转载者" 涉及的主要知识点是使用Vert.x框架处理HTTPS通信中的异常。Vert.x是一个轻量级、反应式、非阻塞的Java库,用于构建分布式系统,特别是网络应用。在这个项目中,...

    Exceptions:javasctipt的基本异常类

    标题中的"Exceptions: JavaScript的基本异常类"指的就是JavaScript中用于表示和处理错误的结构。JavaScript提供了一个内置的异常处理系统,它基于`Error`对象及其子类。这个系统允许开发者在遇到错误时抛出异常,并...

    ocp-ws-exceptions:OCP研讨会-例外

    "ocp-ws-exceptions: OCP研讨会-例外"这个标题暗示我们将会深入探讨Java中的异常处理,特别是与Oracle Certified Professional (OCP)相关的部分。OCP是Java开发者认证的一种,它涵盖了广泛的Java知识,包括异常处理...

    Python 解决execjs._exceptions.ProgramError: ReferenceError: document is not defined报错问题

    js_obj = execjs.compile(js_code, cwd='node_modules') ``` 但这样做可能会遇到新的问题,例如`TypeError: Cannot read property '0' of null`,这通常表示在JavaScript代码中尝试访问一个`null`对象的属性。要...

    mvc-exceptions:我的MVC例外博客中涵盖的要点演示

    该应用程序演示了我的MVC Exceptions博客中涵盖的大多数要点: 。 发行历史 2013年11月:V1 2014年10月:V2 2018年4月:V2.0.1 2021年1月:V2.1.0 应用概述 最重要的文件是: 演示1 带有@ExceptionHand

    epfl-exceptions:例外的定义

    epfl-例外包含epfl软件包的异常用法var exceptions = require ( 'epfl-exceptions' ) ;var ParameterException = exceptions . ParameterException ;var ServerException = exceptions . ServerException ;var ...

    EntityFramework.Exceptions:使用Entity Framework Core时轻松处理数据库错误。 支持SQLServer,PostgreSQL,SQLite,Oracle和MySql

    EntityFramework.Exceptions 使用Entity Framework Core时,轻松处理数据库错误。 支持SQLServer,PostgreSQL,SQLite和MySql EntityFramework.Exceptions有什么作用? 使用Entity Framework Core进行数据访问时,...

    Exceptions:异常处理

    在Java编程语言中,异常处理是一项至关重要的技能,它确保了程序在遇到错误或异常情况时能够优雅地处理问题,而不是突然崩溃。异常是程序执行过程中出现的不正常情况,通常由错误的操作、非法的数据或者不可预见的...

    retry_on_exceptions:用于通过捕获指定的异常之一然后重试来重试函数 N 次的装饰器

    retry_on_exceptions 装饰器用于通过捕获指定的异常之一然后重试来重试函数 N 次的装饰器。 对于偶尔抛出错误的函数特别有用,例如依赖外部资源(如 Web API、数据库等)的函数。 安装: pip install retry_on_...

    http-exceptions:带有错误代码的 HTTP 异常

    this.statusCode = statusCode; this.name = 'HTTPException'; } } ``` 这样的自定义异常可以帮助我们区分不同类型的HTTP错误,并提供额外的状态码信息。 标签"JavaScript"表明这个话题主要是在JavaScript上下...

    Java-CustomExceptions:Java 自定义异常

    Java异常分为两种类型:未检查异常(Unchecked Exceptions)和检查异常(Checked Exceptions)。自定义异常如果继承自`RuntimeException`或其子类,将成为未检查异常,否则默认为检查异常。未检查异常在编译时不强制...

    net系统错误记录工具 Elmah

    十分方便的一个asp.net的日志管理dll。无需写代码,只要配置一下就可以。 是一款ASP.NET下的系统错误记录管理工具,它可以非常方便的把“黄屏”错误记录到XML,MS SQLServer,SQLite,MySql等文件中,甚至它还可以...

    exceptions:mtl友好异常

    例外情况 该软件包提供了(可选的纯)可扩展异常,这些异常与monad转换器库兼容。 联系信息 欢迎提供贡献和错误报告! 请随时通过github或irc.freenode.net上的#haskell IRC频道与我联系。 爱德华·克梅特...

    Cabother-Exceptions:自定义例外的参考书目

    Cabother例外自定义例外的参考书目努吉特https://www.nuget.org/packages/Cabother.Exceptions/例外清单Nome da Exception Descrição Mensagem doUsuário ConfigurationNotFoundException Solicitado ...

Global site tag (gtag.js) - Google Analytics