保密级别:内部中国农业银行资金清算平台软件需求规格说明书-1/10-软件开发中心1.底层篇1.1.基本数据类型时间分析int的运算速度最快,short次之,byte再次之,long再次之。float和double运算速度最慢。除法比乘法慢的太多,基本上除法是乘法的9倍时间。当然,除了浮点型外。根据intelcpu的参考数据,乘法计算时间是移位运算的4-5倍是比较正常的。long类型的计算很慢,建议一般少使用它。double运算速度和float相当;浮点的乘法比除法要慢。但是,这个结果并不能真正说明问题。这个结果只是一个一般性的,在特殊情况下,乘法还是比除法快,比如:floatA*floatB仍然是比floatA/(1/floatB)快。从实际的数据结果来讲,乘法的时候,乘数越小速度越快,特别是在乘数比3小的时候,乘法时耗接近20,大于4的时候,几乎都是600的时耗。除法恰好相反,除数大于1的时候,时耗一般都是350,可是,当除数小于1的时候,时耗就变成了700了。对于大家关心的移位和乘除2的问题,jdk5.0已经做了部分处理。即“var*=2”和“var=1”耗费一样。但是,除法并没有进行这类处理,即“var/=2”耗费和基本的除法一样。1.2.类和接口调用时间分析1.2.1.类的创建虽然面向对象思想已经深入人心,但他在带来快捷方面的编程风格的时候,也带来了低下的效率。在Java中,反应最快的是Object类(这也是显然的),建立一个新的Object类时耗仅仅为20单位。而一个空类(即没有声明任何Methods和Fields)的建立时间则增加到了惊人的400单位。如果再给类增加一些字段的话,时间耗费并没有特别大的增加,每增加一个int类型字段大概增加30个单位。仅仅就创建时间来说,内嵌的类型都有不错的表现。比如,创建一个int数组(仅仅包含一个元素)的时间只比创建一个Object对象的时间多一倍。当然,如果你创建的数组对象包含1000个元素的话,其创建时间显然还会加上内存管理的时间了,它的时间大概是1万个时间单位。请注意,我们这里讨论的时间单位其实十分小,1万个时间单位也仅仅只是有0.006毫秒(0.000006秒)。创建一个byte、short、int、long、float和double数组对象的时间消耗几乎是一样的。1.2.2.方法的调用Java在这个方面有一点做得很好,就是调用一个只有很少量代码的方法的时耗和直接把这段代码写到本地的时耗相差很小。当然不包括需要分配很多本地变量的情况。调用本类(this指针)的方法是最快的,时间在1-2个单位。调用其它类的静态方法也很快,速度和调用本来方法差不多。调用其它类的非静态方法速度就慢一些,在1.5-2.5个时间单位之间。保密级别:内部中国农业银行资金清算平台软件需求规格说明书-2/10-软件开发中心调用继承接口的方法是十分缓慢的,是调用普通方法的3倍。但是,如果在实现接口的时候加上final关键字的话,调用这个方法的时耗就和普通方法差不多了。最慢的是已经同步化了的方法。即加上了synchronized关键字的方法。调用它的时耗比普通方法高出了近20倍。如果不是万不得已,不要把synchronized加到方法上面,实在不行的话,你可以把它加到代码块上去,这种加法比直接加到方法上面快一点。注意,因为方法大部分时候都是完成很多事情的,所以,十分注意调用方法的开销是没有必要的,因为这个时间和方法执行需要的时间比较起来只是毛毛雨。1.3.基本操作时间耗费2.通用篇“通用篇”讨论的问题适合于大多数Java应用。2.1.不用new关键词创建类的实例用new关键词创建类的实例时,构造函数链中的所有构造函数都会被自动调用。但如果一个对象实现了Cloneable接口,我们可以调用它的clone()方法。clone()方法不会调用任何类构造函数。在使用设计模式(DesignPattern)的场合,如果用Factory模式创建对象,则改用clone()方法创建新的对象实例非常简单。例如,下面是Factory模式的一个典型实现:publicstaticCreditgetNewCredit(){returnnewCredit();}改进后的代码使用clone()方法,如下所示:privatestaticCreditBaseCredit=newCredit();publicstaticCreditgetNewCredit(){return(Credit)BaseCredit.clone();}上面的思路对于数组处理同样很有用。保密级别:内部中国农业银行资金清算平台软件需求规格说明书-3/10-软件开发中心2.2.使用非阻塞I/O版本较低的JDK不支持非阻塞I/OAPI。为避免I/O阻塞,一些应用采用了创建大量线程的办法(在较好的情况下,会使用一个缓冲池)。这种技术可以在许多必须支持并发I/O流的应用中见到,如Web服务器、报价和拍卖应用等。然而,创建Java线程需要相当可观的开销。JDK1.4引入了非阻塞的I/O库(java.nio)。如果应用要求使用版本较早的JDK,在这里有一个支持非阻塞I/O的软件包。请参见Sun中国网站的《调整Java的I/O性能》。2.3.慎用异常异常对性能不利。抛出异常首先要创建一个新的对象。Throwable接口的构造函数调用名为fillInStackTrace()的本地(Native)方法,fillInStackTrace()方法检查堆栈,收集调用跟踪信息。只要有异常被抛出,VM就必须调整调用堆栈,因为在处理过程中创建了一个新的对象。异常只能用于错误处理,不应该用来控制程序流程。2.4.不要重复初始化变量默认情况下,调用类的构造函数时,Java会把变量初始化成确定的值:所有的对象被设置成null,整数变量(byte、short、int、long)设置成0,float和double变量设置成0.0,逻辑值设置成false。当一个类从另一个类派生时,这一点尤其应该注意,因为用new关键词创建一个对象时,构造函数链中的所有构造函数都会被自动调用。2.5.尽量使用局部变量调用方法时传递的参数以及在调用中创建的临时变量都保存在栈(Stack)中,速度较快。其他变量,如静态变量、实例变量等,都在堆(Heap)中创建,速度较慢。另外,依赖于具体的编译器/JVM,局部变量还可能得到进一步优化。请参见《尽可能使用堆栈变量》。保密级别:内部中国农业银行资金清算平台软件需求规格说明书-4/10-软件开发中心2.6.位移完成乘法和除法考虑下面的代码:for(val=0;val100000;val+=5){alterX=val*8;myResult=val*2;}用移位操作替代乘法操作可以极大地提高性能。下面是修改后的代码:for(val=0;val100000;val+=5){alterX=val3;myResult=val1;}修改后的代码不再做乘以8的操作,而是改用等价的左移3位操作,每左移1位相当于乘以2。相应地,右移1位操作相当于除以2。值得一提的是,虽然移位操作速度快,但可能使代码比较难于理解,所以最好加上一些注释。2.7.避免在循环体中创建对象,即使该对象占用内存空间不大.for(inti=0;i10000;++i){Objectobj=newObject();System.out.println(obj=+obj);}应改成Objectobj=null;for(inti=0;i10000;++i){obj=newObject();System.out.println(obj=+obj);}保密级别:内部中国农业银行资金清算平台软件需求规格说明书-5/10-软件开发中心2.8.当做数组拷贝操作时,采用System.arraycopy()方法2.9.尽量避免在循环体中调用方法2.10.尽量避免在循环体中使用try-catch块2.11.在多重循环中,如果有可能,尽量将最长的循环放在最内层,最短的循环放在最外层,以减少循环层间的变换次数2.12.如果预知长度,就设置ArrayList的长度2.13.避免在类在构造器的初始化其他类2.14.尽量避免在构造中对静态变量做赋值操作2.15.不要在类的构造器中创建类的实例3.JSP/servlet篇3.1.采用out对象中的print方法代替println()方法3.2.采用适当的值初始化out对象缓冲区的大小3.3.尽量采用forward()方法重定向新的JSP3.4.通过init()方法来缓存一些静态数据以提高应用性能.3.5.ServletOutputStream取代PrintWriter4.DB篇保密级别:内部中国农业银行资金清算平台软件需求规格说明书-6/10-软件开发中心4.1.纯JDBC最快4.2.最重要的是尽量减少与数据库通信的次数,多使用批处理4.3.设置合适的Fetch_Size和Batch_Size4.4.采用连接池技术4.5.选择合适的事务隔离层与及时关闭连接对象4.6.PreparedStatement防止sql注入4.7.尽可能地做批处理更新5.内存篇5.1.别用newBoolean()在很多场景中Boolean类型是必须的,比如JDBC中boolean类型的set与get都是通过Boolean封装传递的,大部分ORM也是用Boolean来封装boolean类型的,比如:ps.setBoolean(isClosed,newBoolean(true));ps.setBoolean(isClosed,newBoolean(isClosed));ps.setBoolean(isClosed,newBoolean(i==3));通常这些系统中构造的Boolean实例的个数是相当多的,所以系统中充满了大量Boolean实例小对象,这是相当消耗内存的。Boolean类实际上只要两个实例就够了,一个true的实例,一个false的实例。Boolean类提供两了个静态变量:publicstaticfinalBooleanTRUE=newBoolean(true);publicstaticfinalBooleanFALSE=newBoolean(false);需要的时候只要取这两个变量就可以了,比如:保密级别:内部中国农业银行资金清算平台软件需求规格说明书-7/10-软件开发中心ps.setBoolean(isClosed,Boolean.TRUE);那么象2、3句那样要根据一个boolean变量来创建一个Boolean怎么办呢?可以使用Boolean提供的静态方法:Boolean.valueOf()比如:ps.setBoolean(isClosed,Boolean.valueOf(isClosed));ps.setBoolean(isClosed,Boolean.valueOf(i==3));因为valueOf的内部实现是:return(b?TRUE:FALSE);所以可以节省大量内存。相信如果Java规范直接把Boolean的构造函数规定成private,就再也不会出现这种情况了。5.2.别用newInteger和Boolean类似,java开发中使用Integer封装int的场合也非常多,并且通常用int表示的数值通常都非常小。SUNSDK中对Integer的实例化进行了优化,Integer类缓存了-128到127这256个状态的Integer,如果使用Integer.valueOf(inti),传入的int范围正好在此内,就返回静态实例。这样如果我们使用Integer.valueOf代替newInteger的话也将大大降低内存的占用。如果您的系统要在不同的SDK(比如IBMSDK)中使用的话,那么可以自己做了工具类封装一下,比如IntegerUtils.valueOf(),这样就可以在任何SDK中都可以使用这种特性。5.3.用StringBuffer代替字符串相加这个我就不多讲了,因为已经被人讲过