Java异常的性能分析
详见:http://blog.yemou.net/article/query/info/tytfjhfascvhzxcyt276
在Java中抛异常的性能是非常差的。通常来说,抛一个异常大概会消耗100到1000个时钟节拍。
通常是出现了意想不到的错误,我们才会往外抛异常。也就是说,我们肯定不希望一个进程一秒钟就抛出上千个异常。不过有时候你确实会碰到有些方法把异常当作事件一样往外抛。我们在这篇文章中已经看到一个这样的典范):sun.misc.BASE64Decoder之所以性能很差就是因为它通过抛异常来对外请求道,”我还需要更多的数据“:
| 1 2 3 4 5 6 7 8 9 10 | at java.lang.Throwable.fillInStackTrace(Throwable.java:-1)at java.lang.Throwable.fillInStackTrace(Throwable.java:782)- locked <0x6c> (a sun.misc.CEStreamExhausted)at java.lang.Throwable.<init>(Throwable.java:250)at java.lang.Exception.<init>(Exception.java:54)at java.io.IOException.<init>(IOException.java:47)at sun.misc.CEStreamExhausted.<init>(CEStreamExhausted.java:30)at sun.misc.BASE64Decoder.decodeAtom(BASE64Decoder.java:117)at sun.misc.CharacterDecoder.decodeBuffer(CharacterDecoder.java:163)at sun.misc.CharacterDecoder.decodeBuffer(CharacterDecoder.java:194) | 
如果你用一个数字开头,字母结尾的字符串来运行下这篇文章里面的pack方法,你也会碰到类似的情况。我们来看下用那个方法打包"12345"和"12345a"需要多长的时间:
| 1 2 | Made 100.000.000 iterations forstring '12345': time = 12.109 secMade 1.000.000 iterations forstring '12345a': time = 21.764 sec | 
可以看到,’12345a'迭代的次数要比‘12345’少100倍。也就是说这个方法处理'12345a'慢了差不多200倍。大多数的处理时间都在填充异常的栈跟踪信息了:
| 1 2 3 4 5 6 7 8 9 10 11 12 13 14 | at java.lang.Throwable.fillInStackTrace(Throwable.java:-1)        at java.lang.Throwable.fillInStackTrace(Throwable.java:782)        - locked <0x87> (a java.lang.NumberFormatException)        at java.lang.Throwable.<init>(Throwable.java:265)        at java.lang.Exception.<init>(Exception.java:66)        at java.lang.RuntimeException.<init>(RuntimeException.java:62)        at java.lang.IllegalArgumentException.<init>(IllegalArgumentException.java:53)        at java.lang.NumberFormatException.<init>(NumberFormatException.java:55)        at java.lang.NumberFormatException.forInputString(NumberFormatException.java:65)        at java.lang.Long.parseLong(Long.java:441)        at java.lang.Long.valueOf(Long.java:540)        at com.mvorontsov.javaperf.StrConvTests.pack(StrConvTests.java:69)        at com.mvorontsov.javaperf.StrConvTests.test(StrConvTests.java:38)        at com.mvorontsov.javaperf.StrConvTests.main(StrConvTests.java:29) | 
通过手动解析数字,我们可以很容易提升pack方法的性能。不过不要忘了——不到万不得已,不要随便优化。如果你只是解析几个输入参数而已—— keep it simple,就用JDK的方法就好了。如果你要解析大量的消息,又必须调用一个类似pack这样的方法——那确实得去优化一下了。
新的pack方法和旧的实现差不太多——把一个字符串转化成一个尽可能小的Character/Integer/Long/Double/String类型,使得result.toString().equals(orginalString)为true。
| 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 | public static Object strToObject( final String str ){    if( str == null|| str.length() > 17 )    {  //out of Long range        returnstr;    }    if( str.equals( "") )        return""; //ensure interned string is returned    if( str.length() == 1 )        returnstr.charAt( 0 ); //return Character    //if starts with zero - support only "0" and "0.something"    if( str.charAt( 0 ) == '0')    {        if( str.equals( "0") )            return0;        if( !str.startsWith( "0.") )  //this may be a double            returnstr;    }     long res = 0;    int sign = 1;    for( int i = 0; i < str.length(); ++i )    {        final char c = str.charAt( i );        if( c <= '9'&& c >= '0')            res = res * 10 + ( c - '0');        elseif( c == '.')        {            //too lazy to write a proper Double parser, use JDK one            try            {                final Double val = Double.valueOf( str );                //check if value converted back to string equals to an original string                final String reverted = val.toString();                returnreverted.equals( str ) ? val : str;            }            catch( NumberFormatException ex )            {                returnstr;            }        }        elseif( c == '-')        {            if( i == 0 )                sign = -1; //switch sign at first position            else                returnstr; //otherwise it is not numeric        }        elseif( c == '+')        {            if( i == 0 )                sign = 1; //sign at first position            else                returnstr; //otherwise it is not numeric        }        else//non-numeric            returnstr;    }    //cast to int if value is in int range    if( res < Integer.MAX_VALUE )        return( int ) res * sign;    //otherwise return Long    returnres * sign;} | 
很惊讶吧,新的方法解析数字比JDK的实现快多了!很大一个原因是因为JDK在解析的最后,调用了一个支持的解析方法,像这样:
public static int parseInt( String s, int radix ) throws NumberFormatException
新的方法和旧的相比(注意方法调用的次数——对于非数字串pack只调用了1百万次,而别的情况能调用到千万级别):
| 1 2 3 4 | Pack: Made 100.000.000 iterations forstring '12345': time = 12.145 secPack: Made 1.000.000 iterations forstring '12345a': time = 23.248 secstrToObject: Made 100.000.000 iterations forstring '12345': time = 6.311 secstrToObject: Made 100.000.000 iterations forstring '12345a': time = 5.807 sec | 
总结
千万不要把异常当成返回码一样用,或者当作可能发生的事件(尤其是和IO无关的方法)往外抛。抛异常的代价太昂贵了,对于一般的方法,至少要慢百倍以上。
如果你每条数据都需要解析,又经常会出现非数值串的时候,尽量不要用Number子类型的parse*/valueOf这些方法。为了性能考虑,你应当手动解析它们。
Java异常的性能分析的更多相关文章
- Java应用常用性能分析工具
		Java应用常用性能分析工具 好的工具有能有效改善和提高工作效率或加速分析问题的进度,笔者将从事Java工作中常用的性能工具和大家分享下,如果感觉有用记得投一票哦,如果你有好的工具也可以分享给我 工具 ... 
- JAVA 异常对于性能的影响
		陶炳哲 - MAY 12, 2015 在对OneAPM的客户做技术支持时,我们常常会看到很多客户根本没意识到的异常.在消除了这些异常之后,代码运行速度与以前相比大幅提升.这让我们产生一种猜测,就是在代 ... 
- Java几款性能分析工具的对比
		在给客户的应用程序维护的过程中,我注意到在高负载下的一些性能问题.理论上,增加对应用程序的负载会使性能等比率的下降.然而,我认为性能下降的比率远远高于负载的增加.我也发现,性能可以通过改变应用程序的逻 ... 
- [转]Java Thread Dump 性能分析
		Java and Thread 一个 web 服务器使用几十到几百个线程来处理大量并发用户,如果一个或多个线程使用相同的资源,线程之间的竞争就不可避免了,并且有时候可能会发生死锁. Thread co ... 
- 使用JDK自带的VisualVM进行Java程序的性能分析
		VisualVM是什么? VisualVM是JDK自带的一个用于Java程序性能分析的工具,JDK安装完毕后就有啦,在JDK安装目录的bin文件夹下能找到名称为jvisualvm.exe. 要使用Vi ... 
- Java 性能分析工具 , 第 1 部分: 操作系统工具
		引言 性能分析的前提是将应用程序内部的运行状况以及应用运行环境的状况以一种可视化的方式更加直接的展现出来,如何来达到这种可视化的展示呢?我们需要配合使用操作系统中集成的程序监控工具和 Java 中内置 ... 
- Java 性能分析工具 , 第 3 部分: Java Mission Control
		引言 本文为 Java 性能分析工具系列文章第三篇,这里将介绍如何使用 Java 任务控制器 Java Mission Control 深入分析 Java 应用程序的性能,为程序开发人员在使用 Jav ... 
- 软件性能测试分析与调优实践之路-Java应用程序的性能分析与调优-手稿节选
		Java编程语言自从诞生起,就成为了一门非常流行的编程语言,覆盖了互联网.安卓应用.后端应用.大数据等很多技术领域,因此Java应用程序的性能分析和调优也是一门非常重要的课题.Java应用程序的性能直 ... 
- Java 性能分析工具 , 第 2 部分:Java 内置监控工具
		引言 本文为 Java 性能分析工具系列文章第二篇,第一篇:操作系统工具.在本文中将介绍如何使用 Java 内置监控工具更加深入的了解 Java 应用程序和 JVM 本身.在 JDK 中有许多内置的工 ... 
随机推荐
- ECMAScript6新特性之let、const
			第一次在博客园写博客,想把自己每一天学习到的知识点记录下来,心里有点紧张(PS:不知道自己能不能写好......嘿嘿).言归正传,咱们先来说说"ECMAScript"这到底是啥玩意 ... 
- Java ee 与安卓环境搭建个人心得
			最近加了个IT俱乐部,第一次作业就是搞定eclipse,完成Java ee 与安卓环境搭建.为此我上网看了好多教程,之前我安装了Java,可以说省了不少事,而且还了解一点安装方法.流程网上都有,但是不 ... 
- curl---一款实用的URL命令行网络通讯工具/库
			最近一段时间在看朴灵翻译的<深入浅出nodejs>,里面有提到一种脱离浏览器的客户端网络通讯工具,curl命令,自己在电脑上试了一下,感觉非常好用,而且莫名的感觉这是一个非常强大的网络工具 ... 
- 安徽省2016“京胜杯”程序设计大赛_B_阵前第一功
			阵前第一功 Time Limit: 1000 MS Memory Limit: 65536 KB Total Submissions: 63 Accepted: 29 Description A国每个 ... 
- 如何搭建自己的Maven远程私仓
			1.首先,配置好Maven,jdk等必备环境 2.配置好环境后,下载最新版本的nexus 下载地址:http://www.sonatype.org/nexus/go 3.打开目录nexus-***\b ... 
- 在Eclipse中关联Android API源码
			在Eclipse中快速关联API源码,便于查看类以及方法.方法如下: 1. 在对应的项目文件右键——>properties——>java build path——>libraries ... 
- 如何搭建个人博客网站(Mac)
			一直以为自己记忆力很好,毕业之后才发现,之前需要看一遍就能记住的东西,现在看两三遍才能有印象.而搞技术的,如果不及时的记录下当时的情景,过后很容易就忘记.所以,再次萌生了搭博客.写文章的想法(之前用D ... 
- Java获取精确到秒的时间戳(转)
			1.时间戳简介: 时间戳的定义:通常是一个字符序列,唯一地标识某一刻的时间.数字时间戳技术是数字签名技术一种变种的应用.是指格林威治时间1970年01月01日00时00分00秒(北京时间1970年01 ... 
- 一起来学Go   ---   (go的简介以及环境的安装)
			Go 相信大家,看到这篇文章的时候,已经自己在百度百科了解了go的发展史已经特性,再次我依然....得哔哔叨一会. ^.^ go语言的特性 go语言作为一门静态类型开发语言,与当前的开发语言想必具 ... 
- 读书笔记--C陷阱与缺陷(一)
			要参与C语言项目,于是作者只好重拾C语言(之前都是C++,还是C++方便). 看到大家都推荐看看 C陷阱与缺陷(C traps and pitfalls),于是好奇的开始了这本书的读书之旅. 决定将 ... 
