|
该用户从未签到
|
最近在做应用的性能优化,在review代码的过程中积累了一些规则和经验。做到这些规则的目的很简单,就是写出“优美”的代码来。0 z$ ~; h2 B- `( i- V
, Q# J1 X* g, c* m, r7 H
1、注释尽可能全面
2 s* Z, o7 z' R5 M% F1 _3 M" ? E) a% x
对于方法的注释应该包含详细的入参和结果说明,有异常抛出的情况也要详细叙述;类的注释应该包含类的功能说明、作者和修改者。5 q& f6 | i r4 f: l
5 ^7 b. K' u, O8 L2、多次使用的相同变量最好归纳成常量9 N- K6 k: L0 v5 W+ G! {' ^; g3 w4 @
! f7 G& S \* U多处使用的相同值的变量应该尽量归纳为一个常量,方便日后的维护。6 l; y/ f" q5 N% q$ d; d7 R
! w1 I4 Q0 ?4 |; [) b }
3、尽量少的在循环中执行方法调用( q5 N5 l$ e1 J% j! I% {8 s
4 K# j& ^/ {9 {1 ^( m7 K! k尽量在循环中少做一些可避免的方法调用,这样可以节省方法栈的创建。例如:
, \* W: n! ^% l6 f8 S' R; V" z0 x9 _3 {& Z0 x' y
for(int i=0;i<list.size();i++){ ! C" I$ f7 T) y* s0 Y/ ~ Y
System.out.println(i);
& g$ z* D8 x8 X( `} - [% `7 Y: i: O7 i& `1 H6 F; j
可以修改为:# }$ [2 W0 l' a. b$ G
$ m: S# c8 Y& d3 [9 I0 h
for(int i=0,size=list.size();i<size;i++){ 9 K" q7 z# M& {$ s5 ]
System.out.println(i);
, d+ ]+ s, J2 N" o7 j7 d0 A} & W& T R2 i* n! r2 e" `
4、常量的定义可以放到接口中
5 V& T- h# J. \& ?# d, [0 ?1 w! I! F
在java中,接口里只允许存在常量,因此把常量放到接口中声明就可以省去public static final这几个关键词。
1 C' T& z/ h4 d: p
( ~) n- y& Y8 E( K3 x; k5、ArrayList和LinkedList的选择8 N8 t" k4 H' A, z- y% r8 x: p
# R4 B' h/ t5 K( Z6 E3 g# [这个问题比较常见。通常程序员最好能够对list的使用场景做出评估,然后根据特性作出选择。ArrayList底层是使用数组实现的,因此随机读取数据会比LinkedList快很多,而LinkedList是使用链表实现的,新增和删除数据的速度比ArrayList快不少。
8 @2 v$ O; q* B. |5 T
3 o# z1 f. N' Q& |+ {) Y7 D X& j6、String,StringBuffer和StringBuilder
9 K9 v) F9 c, w/ B5 S) p# J* e: ^0 E( h) m2 L# E
这个问题也比较常见。在进行字符串拼接处理的时候,String通常会产生多个对象,而且将多个值缓存到常量池中。例如:) W8 u- I/ e, x' J. Y$ K5 |% w
1 @0 Q- E4 v _7 @2 q, LString a="a";
) y( G, J8 L4 DString b="b";
4 i2 v: N) U' Q z2 ?, Ma=a+b;
- o6 E' W7 g5 ~( U+ W这种情况下jvm会产生"a","b","ab"三个对象。而且字符串拼接的性能也很低。因此通常需要做字符串处理的时候尽量采用StringBuffer和StringBuilder来。/ N) c( M0 i5 _& I
. n+ w1 o8 _5 d# h( _: o, V
7、包装类和基本类型的选择; \) X* D: c. c$ I2 i6 ?5 O
( k* n* p; M8 I" a3 g在代码中,如果可以使用基本数据类型来做局部变量类型的话尽量使用基本数据类型,因为基本类型的变量是存放在栈中的,包装类的变量是在堆中,栈的操作速度比堆快很多。9 F( a% J2 T6 T
6 j7 X3 T0 R) Y6 [/ C* H1 I8、尽早的将不再使用的变量引用赋给null
+ Z+ n4 t1 B+ w" o, t- ?) a! Z& k% X+ `2 _( z
这样做可以帮助jvm更快的进行内存回收。当然很多人其实对这种做法并不感冒。
* }! F0 v# N \3 j* m
9 s' Q/ p/ Y B: Z6 ~. D9、在finally块中对资源进行释放
3 r( @3 R: @& Z2 w/ A2 I' X% K
$ V1 d: b, W3 ~( j' t% ?- V/ b7 z典型的场景是使用io流的时候,不论是否出现异常最后都应该在finally中对流进行关闭。" ]/ a& k. h" N; k! b( f3 G+ F9 l
, K4 P# B& q2 Q; t
10、在HashMap中使用一个Object作为key时要注意如何区分Object是否相同; u {- [4 K* ]) d: f
6 r6 I \* k* b在jdk的HashMap实现中,判断两个Object类型的key是否相同的标准是hashcode是否相同和equals方法的返回值。如果业务上需要对两个数据相同的内存对象当作不同的key存储到hashmap中就要对hashcode和equals方法进行覆盖。
. i$ n. `6 c* }) `
$ @( v2 R$ ]% R( ` |
|