代码审查是消灭Bug最重要的方法之一,这些审查在大多数时候都特别奏效。由于代码审查本身所针对的对象,就是俯瞰整个代码在测试过程中的问题和Bug。并且,代码审查对消除一些特别细节的错误大有裨益,尤其是那些能够容易在阅读代码的时候发现的错误,这些错误往往不容易通过机器上的测试识别出来。本文就常见的Java代码中容易出现的问题提出一些建设性建议,以便您在审查代码的过程中注意到这些常见的细节性错误。 %+nM4)h
e"UXG\8D
Vm?# ~}T
通常给别人的工作挑错要比找自己的错容易些。别样视角的存在也解释了为什么作者需要编辑,而运动员需要教练的原因。不仅不应当拒绝别人的批评,我们应该欢迎别人来发现并指出我们的编程工作中的不足之处,我们会受益匪浅的。 1`1jSx5}.
a ~YrQI-@
/!J xiGn
cTz@ga;!mI
正规的代码审查(code inspection)是提高代码质量的最强大的技术之一,代码审查?由同事们寻找代码中的错误?所发现的错误与在测试中所发现的错误不同,因此两者的关系是互补的,而非竞争的。 yEMM@5W)8
=),O ;M
P*jiz@6
EKO~\d
如果审查者能够有意识地寻找特定的错误,而不是靠漫无目的的浏览代码来发现错误,那么代码审查的效果会事半功倍。在这篇文章中,我列出了11个Java编程中常见的错误。你可以把这些错误添加到你的代码审查的检查列表(checklist)中,这样在经过代码审查后,你可以确信你的代码中不再存在这类错误了。 GSs?!BIC
V?Q45t Ae
4X",:B}
])G|U A.
一、常见错误1# :多次拷贝字符串 (t){o>l
# >I_
:@@`N_2?
=jKu=!QPq
测试所不能发现的一个错误是生成不可变(immutable)对象的多份拷贝。不可变对象是不可改变的,因此不需要拷贝它。最常用的不可变对象是String。 _u""v
E,F^!4 rJ$
)3A+Ell`
#D#kw*c
如果你必须改变一个String对象的内容,你应该使用StringBuffer。下面的代码会正常工作: C?k\5AzT
amq,^
=`*@OJHH
>0[:uu,'>
String s = new String ("Text here"); KwV!smi2
}9^'etD
B uso
`G
=E$bZe8
但是,这段代码性能差,而且没有必要这么复杂。你还可以用以下的方式来重写上面的代码: A9g/At_
p0y|pD
$tF\7.e@
~3-"1E>Rgy
String temp = "Text here"; RX%)@e/@
String s = new String (temp); nGwon8&]]
U.V/JbXX
*P5\T4!+d
O8A(OfX
但是这段代码包含额外的String,并非完全必要。更好的代码为: (,ik:j
V;g) P
-+u}u=z%
$9j>oUG
String s = "Text here"; |Xm$O1Wa
S,C c0)j>
JU;`c>8=)
@ ;@~=w
二、常见错误2#: 没有克隆(clone)返回的对象 p^}`^>OL
$a8,C\me?
3M(*q4A$"
k q]E@tE*3
封装(encapsulation)是面向对象编程的重要概念。不幸的是,Java为不小心打破封装提供了方便??Java允许返回私有数据的引用(reference)。下面的代码揭示了这一点: {]U
\HE1w
GqNOWK2O
"+4Jmf9
t9MCT$U
import java.awt.Dimension; 9D &vxKE
/***Example class.The x and y values should never*be negative.*/ *59|
public class Example{ */JYP +
private Dimension d = new Dimension (0, 0); l*]hUP J
public Example (){ } _;0RW
gvc/Z <Y
/*** Set height and width. Both height and width must be nonnegative * or an exception is thrown.*/ +}1zw<
public synchronized void setValues (int height,int width) throws IllegalArgumentException{ mI{Fs|9h
if (height < 0 || width < 0) M%la@2SK=
throw new IllegalArgumentException(); l53Q"ajG
d.height = height; -9.lFuI
d.width = width; $j(d`@.DN~
} hr&&b3W3p
DA iS|x
public synchronized Dimension getValues(){ <,0/BMz
// Ooops! Breaks encapsulation jjQDw=6
return d; q9p31b3
} M9o/6
} oK-d58 sM
X `EVjK
bM5V=b_H
nSh~mP
Example类保证了它所存储的height和width值永远非负数,试图使用setValues()方法来设置负值会触发异常。不幸的是,由于getValues()返回d的引用,而不是d的拷贝,你可以编写如下的破坏性代码: J_7@d]0R
[&4+
<Nl'
'_V9FWDZ
]" e'z
Example ex = new Example(); KQb&7k.
Dimension d = ex.getValues(); MRXw)NAw
d.height = -5; >q&5Z
d.width = -10; ^n<YO=|u
U^|T{g+O
o~e_M-
]T|$nwQ
现在,Example对象拥有负值了!如果getValues() 的调用者永远也不设置返回的Dimension对象的width 和height值,那么仅凭测试是不可能检测到这类的错误。 ;-JF b$m
!ht2*8$lQ
E:M,nSc)53
]\ !ka/%
不幸的是,随着时间的推移,客户代码可能会改变返回的Dimension对象的值,这个时候,追寻错误的根源是件枯燥且费时的事情,尤其是在多线程环境中。 /*>}y$
YmFg#eS
9xj }<WM
g 8uq6U
更好的方式是让getValues()返回拷贝: j0X^,ot@m
F .Zk};lb
Z3YKG{g
kaQNcMcq
public synchronized Dimension getValues(){ boCi*]
return new Dimension (d.x, d.y); 2A@oa9
} 5@r6'Z
u-y?i`
7FPSBvU#/
4)OOj14-V
现在,Example对象的内部状态就安全了。调用者可以根据需要改变它所得到的拷贝的状态,但是要修改Example对象的内部状态,必须通过setValues()才可以。 %XeN_
V
Jn d_cJ ]a
.tGz, z}
gED|2%BXb
三、常见错误3#:不必要的克隆 1\UU"
RCYv 2=m>Q
6nE/8m
?D2a"a$^
我们现在知道了get方法应该返回内部数据对象的拷贝,而不是引用。但是,事情没有绝对: .%7Le|Fb"
g(X`.0
{DKZ~
)-1e}VF(U
/*** Example class.The value should never * be negative.*/ \-]tvgA~&
public class Example{ n.a2%,|v
private Integer i = new Integer (0); a%U#PF6
public Example (){ } 6,jCO@!
1eV&oN#
/*** Set x. x must be nonnegative* or an exception will be thrown*/ gJuK% P
public synchronized void setValues (int x) throws IllegalArgumentException{ ?B;7J7 T
if (x < 0) Q|{b8K
throw new IllegalArgumentException(); m:`M&Xs&
i = new Integer (x); - E GZ
} %X.g+uu
{wA8!5Gu
public synchronized Integer getValue(){ w0Nm.=I-
// We can’t clone Integers so we makea copy this way. ,D*bLXWh
return new Integer (i.intValue()); xR%NiYNQz
} [^ r8P:Ad
} >itabG-&
zI,Qc60B
13Z,;YW
HyWR&0J
这段代码是安全的,但是就象在错误1#那样,又作了多余的工作。Integer对象,就象String对象那样,一旦被创建就是不可变的。因此,返回内部Integer对象,而不是它的拷贝,也是安全的。 O9d"Z$~n=j
<`=Kt[_BQ
P2f^]z
UCmy$aW
方法getValue()应该被写为: pO7OP"q1
vX6JjE!
gsEcvkj*
LFxk.-{=
public synchronized Integer getValue(){ \ +sa[jK
// ’i’ is immutable, so it is safe to return it instead of a copy. ;A@DE@^5w
return i; (M1YOK) I
} M_UmnqN1C
bri8o"
*A8*FX>\F
&}Wi@;G]2
Java程序比C++程序包含更多的不可变对象。JDK 所提供的若干不可变类包括: 6@/k|t>OT
7- LjBlH
\/j,
s+fxv(,"c
?Boolean R!"|~OO
?Byte ,9jk<)m]L
?Character p&Qm[!
?Class `5h^!="
?Double ZAy/u@qt
?Float \db=]L=|
?Integer %5zIh[!1$
?Long @w.DN)GPo
?Short L>1y[
Q
?String 56c[$ q
?大部分的Exception的子类 5vR])T/S0
+:ms`Sr>
w.J$(o(/
L)\<7
四、常见错误4# :自编代码来拷贝数组 'Z.C&6_
F5YoEWS
?yjg\S?L
^&mrY[;S
Java允许你克隆数组,但是开发者通常会错误地编写如下的代码,问题在于如下的循环用三行做的事情,如果采用Object的clone方法用一行就可以完成: H.>EO|p
vxk0@k_
# }}6JM
r^msJ|k8[
public class Example{ Hc>yZ:c;
private int[] copy; @|t]9
/*** Save a copy of ’data’. ’data’ cannot be null.*/ GXD<X_[
public void saveCopy (int[] data){ sUc[!S:/
copy = new int[data.length]; fa/o4S<
for (int i = 0; i < copy.length; ++i) ^{=UKf{
copy = data; V[*>}XQER
} 9Xa.%vw>
} . 70=xH
W/t,7lPFb
c u";rnj
,9I-3**W
这段代码是正确的,但却不必要地复杂。saveCopy()的一个更好的实现是: Twd*HH
+HUy,@^Pa
B/@LE{qUn
)SZ#%OE*
void saveCopy (int[] data){ 2SlL`hN>Z
try{ ]9w8[T:O
copy = (int[])data.clone(); p9 ,[kb
}catch (CloneNotSupportedException e){ XY? Cl
// Can’t get here. fB7Jx6
} MS#*3Md&y
} VO {z)_
oGI'a:iff
*BM#fe
ackeq#
如果你经常克隆数组,编写如下的一个工具方法会是个好主意: s1::\&`za
)i:*r8*~
k\SqDmv
UNiK6h_%
static int[] cloneArray (int[] data){ S!$S'{f<
try{ y5aPs z
return(int[])data.clone(); pT~3<
,
}catch(CloneNotSupportedException e){ Z+6WG
// Can’t get here. 5HHf3E [
} )hQ]>o@i{
} #*y.C[^5{
YCb|eS^u
z(%tu
=ca[*0^Z7
这样的话,我们的saveCopy看起来就更简洁了: y O@1#
??.aLeF&
8`)* ?Q9~
0n2H7}Uq
void saveCopy (int[] data){ Gukvd6-g9b
copy = cloneArray ( data); hPz=Ec<zW
} xgkCN$zQ`
y*sVimx
pnp8`\cIH
M"p%CbcI]
五、常见错误5#:拷贝错误的数据 Pke8RLg2A
oO3^9?Z
q&9]4j
C|IHRw`[
有时候程序员知道必须返回一个拷贝,但是却不小心拷贝了错误的数据。由于仅仅做了部分的数据拷贝工作,下面的代码与程序员的意图有偏差: "bRjY?D
?#&[1.= u
(vD==n9Hd
\P":V
import java.awt.Dimension; 0iR?r+|
/*** Example class. The height and width values should never * be 3[_WTwX0
negative. */ J> ,w},`
public class Example{ VrfEa d
static final public int TOTAL_VALUES = 10; ?Q"<AL>Z
private Dimension[] d = new Dimension[TOTAL_VALUES]; cc`u{F9
public Example (){ } /&47qU4PJ
wVI_SQ<8V
/*** Set height and width. Both height and width must be nonnegative * or an exception will be thrown. */ 4B[pQlg
public synchronized void setValues (int index, int height, int width) throws IllegalArgumentException{ +eH`mI0f
if (height < 0 || width < 0) n<FUaR>q}
throw new IllegalArgumentException(); ZQ`4'|"
if (d[index] == null) r
20!
d[index] = new Dimension(); 90iveb21}
d[index].height = height; jxm#4
d[index].width = width; MxX)&327
} kiyKL:6D|
public synchronized Dimension[] getValues() [hot,\+f
throws CloneNotSupportedException{ <wFmfrx+v
return (Dimension[])d.clone(); `DSFaBj,
} gs i2
} ,/V~T<FI
pnx^a}|px
tQT<1Q02i
baTd;`Pn
这儿的问题在于getValues()方法仅仅克隆了数组,而没有克隆数组中包含的Dimension对象,因此,虽然调用者无法改变内部的数组使其元素指向不同的Dimension对象,但是调用者却可以改变内部的数组元素(也就是Dimension对象)的内容。方法getValues()的更好版本为: lg
)xQV
tzgaHN
%rlqq*
kxyOe[7 S
public synchronized Dimension[] getValues() throws CloneNotSupportedException{ 8q6Le{G
Dimension[] copy = (Dimension[])d.clone(); $\]Mvd
for (int i = 0; i < copy.length; ++i){ q^^R|X1
// NOTE: Dimension isn’t cloneable. m;xa}b{(i
if (d != null) v)|a}5={
copy = new Dimension (d.height, d.width); xfX|AC
} T1Z*>(M
return copy; Glx{Zu=
} OKau3T]
Y^d#8^cP
'
i5}`\
bcuUej:
在克隆原子类型数据的多维数组的时候,也会犯类似的错误。原子类型包括int,float等。简单的克隆int型的一维数组是正确的,如下所示: VFnxj52<
jg%mWiKwK7
Oi~Dio_?
@44*<!da
public void store (int[] data) throws CloneNotSupportedException{ jG& 8`*|*
this.data = (int[])data.clone(); P<[)
qq@;
// OK T 8]*bw
} kt_O=
%cDTq&Q
Si23w'T
7tJ#0to
拷贝int型的二维数组更复杂些。Java没有int型的二维数组,因此一个int型的二维数组实际上是一个这样的一维数组:它的类型为int[]。简单的克隆int[][]型的数组会犯与上面例子中getValues()方法第一版本同样的错误,因此应该避免这么做。下面的例子演示了在克隆int型二维数组时错误的和正确的做法: KdZ=g ZSH
GeB-4img
KX!/n`2u
(Lj*FXmz
public void wrongStore (int[][] data) throws CloneNotSupportedException{ !J {[XT
this.data = (int[][])data.clone(); // Not OK! vg X7B4
} z$g__q-
public void rightStore (int[][] data){ y!S:d
// OK! = 4|"<8'
this.data = (int[][])data.clone();
!P=L0A`
for (int i = 0; i < data.length; ++i){ 'ju_l)(R
if (data != null) 5oB#{h
this.data = (int[])data.clone(); +5R8mbD!
} n) HV:8j~
} 4XiQ8"C
%Y#W#G
q`z1ht
nf
&E!m(|6?+
$5\sV4 8f
六、常见错误6#:检查new 操作的结果是否为null ~K|ha26W
bYhG`1,$-a
Y![i=/
zt|1tU:
Java编程新手有时候会检查new操作的结果是否为null。可能的检查代码为: tOk=m'aUK
Abmi=]\bx
)`W|J%w+
MX!N?k#KhP
Integer i = new Integer (400); ;<0~^,Xm
if (i == null) xa?auv!
throw new NullPointerException(); e_rEu'[av
/yUKUXi
/9D
mK%d
(&V*~OR
检查当然没什么错误,但却不必要,if和throw这两行代码完全是浪费,他们的唯一功用是让整个程序更臃肿,运行更慢。 tv`c"Pb
z([HGq5
,*x/L?.Z!
LKZ<\%
X
C/C++程序员在开始写java程序的时候常常会这么做,这是由于检查C中malloc()的返回结果是必要的,不这样做就可能产生错误。检查C++中new操作的结果可能是一个好的编程行为,这依赖于异常是否被使能(许多编译器允许异常被禁止,在这种情况下new操作失败就会返回null)。在java 中,new 操作不允许返回null,如果真的返回null,很可能是虚拟机崩溃了,这时候即便检查返回结果也无济于事。 %|R]nB
6y?uH;SL
七、常见错误7#:用== 替代.equals ehe#"exCB
n1R{[\ >1
在Java中,有两种方式检查两个数据是否相等:通过使用==操作符,或者使用所有对象都实现的.equals方法。原子类型(int, flosat, char 等)不是对象,因此他们只能使用==操作符,如下所示: S&cN+r
5yV>-XT+-
mQU t 'j4
.]<iRf[\[
int x = 4; Gcxz$.(
int y = 5; M#8_Qbvfk
if (x == y) JH2-'
System.out.println ("Hi"); s{Y-Vdx
// This ’if’ test won’t compile. DmB?.l-
if (x.equals (y)) hS%oQ)zvE
System.out.println ("Hi"); lPA}06hU
Ts=TaRwWf
\qG` ts
CA$|3m9)NM
对象更复杂些,==操作符检查两个引用是否指向同一个对象,而equals方法则实现更专门的相等性检查。 X6r<#n|l
zY4y]k8D*
(X2[}K
XA69t2J~F
更显得混乱的是由java.lang.Object 所提供的缺省的equals方法的实现使用==来简单的判断被比较的两个对象是否为同一个。 Ne1W!0YLK
aE:$ N#|Qa
Wn2J]BH
jEP'jib%
许多类覆盖了缺省的equals方法以便更有用些,比如String类,它的equals方法检查两个String对象是否包含同样的字符串,而Integer的equals方法检查所包含的int值是否相等。 =6fJUy^M\
H:z<]Rc
M4d4b
:V)=/mR
大部分时候,在检查两个对象是否相等的时候你应该使用equals方法,而对于原子类型的数据,你用该使用==操作符。 nU2w\(3|
AuB BSk8($
00Ye
]j_
9r8bSV3`
八、常见错误8#: 混淆原子操作和非原子操作 a?W<<9]
'8@4FXK
^O"o-3dte
v//Drj
Java保证读和写32位数或者更小的值是原子操作,也就是说可以在一步完成,因而不可能被打断,因此这样的读和写不需要同步。以下的代码是线程安全(thread safe)的: `'bu8JK
1u }2}c|
uXG$YDKqC
sbhUW>%.
public class Example{ C,<FV+r=^
private int value; // More code here... uCWBM
public void set (int x){ hb)83mH}
// NOTE: No synchronized keyword u+pZ<Bb
this.value = x; kidv^`.H$w
} /Hq#!2)
} b0N7[M1Xl
9wC='
IG:CWPU
qUQP.4Z9 5
不过,这个保证仅限于读和写,下面的代码不是线程安全的: '|&?$g(\h
r|953e
SmAF+d
2aUE<@RU[
public void increment (){ dA(+02U/.
// This is effectively two or three instructions: ,LU|WXRB
// 1) Read current setting of ’value’. k/Ao?R=@gI
// 2) Increment that setting. Y5mk*Q#q
// 3) Write the new setting back. WBD"d<>'
++this.value; > IZ$ .-
} `n`HwDo;i
2kFP;7FO
E@Yq2FBpnn
ZYTBc#f
在测试的时候,你可能不会捕获到这个错误。首先,测试与线程有关的错误是很难的,而且很耗时间。其次,在有些机器上,这些代码可能会被翻译成一条指令,因此工作正常,只有当在其它的虚拟机上测试的时候这个错误才可能显现。因此最好在开始的时候就正确地同步代码: 7;sF0oB5e
^|cax|>
4%SA%]a L1
}$3pS:_N~
public synchronized void increment (){ \LM{.gzT
++this.value; .;:dG
} "haJwV6-
a{kLAx[>
Z?."cuTt
U\"FYTC
九、常见错误9#:在catch 块中作清除工作 v dU)
ofCN[u
pE G!j ~
Tx$bg(
一段在catch块中作清除工作的代码如下所示: ,esUls'nz'
[O3)s] |
z{U^j:A
% )}rQqQ
OutputStream os = null; (/_w23rr
try{ )u=a+T
os = new OutputStream (); /jn0Xh
// Do something with os here. [Lid%2O3ZR
os.close(); 9_%??@^>
}catch (Exception e){ i6:O9Km
if (os != null) 7{OD/*|
os.close(); a#/~rNRY
} 6lQP+! EF
RJD(c#r$
ooN?x31
>#5jO9
尽管这段代码在几个方面都是有问题的,但是在测试中很容易漏掉这个错误。下面列出了这段代码所存在的三个问题: mk3,ke8
}FkF1?C
:-T[)Q+-3
+,4u1`c|$
1.语句os.close()在两处出现,多此一举,而且会带来维护方面的麻烦。 ^
`[T0X
QM=Y}
'#612iZo
A+"'8%o9}
2.上面的代码仅仅处理了Exception,而没有涉及到Error。但是当try块运行出现了Error,流也应该被关闭。 Es1T{<G|w
*HQ>tvUh
D[K!xq
edfb7prfTl
3.close()可能会抛出异常。 mfgUf
lnrs4s Km
SJ&+"S&
S@WT;Q2Z
上面代码的一个更优版本为: z3|5E#m
`t]8 [P5
Lr(My3vF8q
*V@t]d$=#
OutputStream os = null; %$+bO/f
try{ 3s,a%GOk
os = new OutputStream (); FOSC#W9E
// Do something with os here. " 8g\UR"[
}finally{ ]
N7(<EV/
if (os != null) eeOG(@@o(
os.close(); M4L<u,\1s
} yOm#c>X
sbq:8P#
?#/~BZR!
tr%VYc|}
这个版本消除了上面所提到的两个问题:代码不再重复,Error也可以被正确处理了。但是没有好的方法来处理第三个问题,也许最好的方法是把close()语句单独放在一个try/catch块中。 "0?"
E\
207h$a,
T2ZN=)xZ1
|h2=9\:]
十、常见错误10#: 增加不必要的catch 块 81S0: =
L&Pj0K-HT3
-dH]_
V`"Cd?R0Z
一些开发者听到try/catch块这个名字后,就会想当然的以为所有的try块必须要有与之匹配的catch块。 d+IN-lR(
0@}:`OynX
<*db%{
idY
Xv)R
C++程序员尤其是会这样想,因为在C++中不存在finally块的概念,而且try块存在的唯一理由只不过是为了与catch块相配对。 mS
&^xWPV
VJ-To}
M HKnHPv
f(*iagEy
增加不必要的catch块的代码就象下面的样子,捕获到的异常又立即被抛出: <-=g)3_
tjcG^m} _
{[r}gS%
,TQ;DxB}=E
try{ g"X!&$&
// Nifty code here O7zj8
}catch(Exception e){ ?q}:ojrs1
throw e; }_9yemP
}finally{ {:`XhPS<B
// Cleanup code here '],G!U(
} ;b0;66C8|
)bK3%>H#
}ykc
AK3U
;1Q@d
不必要的catch块被删除后,上面的代码就缩短为: X"Q\MLy
$&.
rS.*
c- "#
W$Z8AZ{E
try{ .-.b:gdO(
// Nifty code here CWS]821;
}finally{ cjf_,x
// Cleanup code here Kq}-)
} kFQx7m
E[>A# l53
x{,W<oXg
FtybF
常见错误11#;没有正确实现equals,hashCode,或者clone 等方法 -}"nb-RR\
N=c{@h
!w:pb7+G
E#c9n%E\sz
方法equals,hashCode,和clone 由java.lang.Object提供的缺省实现是正确的。不幸地是,这些缺省实现在大部分时候毫无用处,因此许多类覆盖其中的若干个方法以提供更有用的功能。但是,问题又来了,当继承一个覆盖了若干个这些方法的父类的时候,子类通常也需要覆盖这些方法。在进行代码审查时,应该确保如果父类实现了equals,hashCode,或者clone等方法,那么子类也必须正确。正确的实现equals,hashCode,和clone需要一些技巧。 D]+@pKb
dp*E#XCr1
6MelN^\[7
Q`z2SYz>
小结 9PJnKzQ4
NdM \RD_R
zl)r3#6hW
w,;ox2
我在代码审查的时候至少遇到过一次这些错误,我自己也犯过其中的几个错误。好消息是只要你知道你在找什么错误,那么代码审查就很容易管理,错误也很容易被发现和修改。即便你找不到时间来进行正规的代码审查,以自审的方式把这些错误从你的代码中根除会大大节省你的调试时间。花时间在代码审查上是值得的。 $qM&iI-l0
]1|OQYG