代码审查是消灭Bug最重要的方法之一,这些审查在大多数时候都特别奏效。由于代码审查本身所针对的对象,就是俯瞰整个代码在测试过程中的问题和Bug。并且,代码审查对消除一些特别细节的错误大有裨益,尤其是那些能够容易在阅读代码的时候发现的错误,这些错误往往不容易通过机器上的测试识别出来。本文就常见的Java代码中容易出现的问题提出一些建设性建议,以便您在审查代码的过程中注意到这些常见的细节性错误。 PRf\6
8KdcLN@
[ Xo
J7
通常给别人的工作挑错要比找自己的错容易些。别样视角的存在也解释了为什么作者需要编辑,而运动员需要教练的原因。不仅不应当拒绝别人的批评,我们应该欢迎别人来发现并指出我们的编程工作中的不足之处,我们会受益匪浅的。 7[?}kG
jQwg)E+o;
J`D<
Rx"VscB6z
正规的代码审查(code inspection)是提高代码质量的最强大的技术之一,代码审查?由同事们寻找代码中的错误?所发现的错误与在测试中所发现的错误不同,因此两者的关系是互补的,而非竞争的。 Sm I8&c
}y=n#%|i.
$MVeMgPa
E6&uZr
如果审查者能够有意识地寻找特定的错误,而不是靠漫无目的的浏览代码来发现错误,那么代码审查的效果会事半功倍。在这篇文章中,我列出了11个Java编程中常见的错误。你可以把这些错误添加到你的代码审查的检查列表(checklist)中,这样在经过代码审查后,你可以确信你的代码中不再存在这类错误了。 [?r\b
0#[Nfe*
FVLA^$5c
apWrcaj
一、常见错误1# :多次拷贝字符串 w!Lb;4x ?
'~6CGqU*
AxqTPx7`|
58gt*yVu
测试所不能发现的一个错误是生成不可变(immutable)对象的多份拷贝。不可变对象是不可改变的,因此不需要拷贝它。最常用的不可变对象是String。 |c+N)FB
K]azUK7
E6IL,Iq9
o>k-~v7
如果你必须改变一个String对象的内容,你应该使用StringBuffer。下面的代码会正常工作: l[j0(T
vH:+
n#AH@`&i
Fl(ZKpSZU
String s = new String ("Text here"); 9*Mg<P"
gmH0-W)=
"%iR-s_>
#[rFep
但是,这段代码性能差,而且没有必要这么复杂。你还可以用以下的方式来重写上面的代码: i*rv_G|(Zj
sU3V)7"
]tV{#iIJ*
+?eAaC7s
String temp = "Text here"; B'~i Z65
String s = new String (temp); )uJ`E8>-
97
X60<
BHBR_7
`-e}:9~q
但是这段代码包含额外的String,并非完全必要。更好的代码为: R_&V.\e_
.:Xe* Q
;Cpm3at
t#Yh!L6>
String s = "Text here"; =7Gi4X%
Zl/+HU~
X7!A(q+h
Z?[J_[ZtR3
二、常见错误2#: 没有克隆(clone)返回的对象 ;Awzm )Q
,=yIfbFQ
8#3cmpx4
6EGEwx
封装(encapsulation)是面向对象编程的重要概念。不幸的是,Java为不小心打破封装提供了方便??Java允许返回私有数据的引用(reference)。下面的代码揭示了这一点: h.%Qn vL
:7]R2JP
bJ[1'Es`
R5~m"bE
import java.awt.Dimension; a.ME{:a%
/***Example class.The x and y values should never*be negative.*/ ;L{y3CWT
public class Example{ dTNgrW`4
private Dimension d = new Dimension (0, 0); sMo%Ayes
public Example (){ } dz DssAHy
?["ZEa
/*** Set height and width. Both height and width must be nonnegative * or an exception is thrown.*/ 2 z l
public synchronized void setValues (int height,int width) throws IllegalArgumentException{ ~#xRoBy3
if (height < 0 || width < 0) 6H9]]Unju
throw new IllegalArgumentException(); $:P~21,
d.height = height; iTyApLV
d.width = width; TMs\#
} \%*y+I0>
UY^f|f&
public synchronized Dimension getValues(){ t38T0Ao
// Ooops! Breaks encapsulation MYm6C;o$
return d; ;#Nci%<J\
} |8rJqtf +&
} L 32ki}2
>.A:6
kE|#mI[>
Z/;SR""wa
Example类保证了它所存储的height和width值永远非负数,试图使用setValues()方法来设置负值会触发异常。不幸的是,由于getValues()返回d的引用,而不是d的拷贝,你可以编写如下的破坏性代码: +d15a%^`
M>#S
z
(/BkwbJyE
S]{Z_|h*j
Example ex = new Example(); HyMb-Us
Dimension d = ex.getValues(); WNi<|A#T{
d.height = -5; &ICO{#v5
d.width = -10; |0
VP^md
EtG)2)
Plv+ mb
w[_Uv4M
现在,Example对象拥有负值了!如果getValues() 的调用者永远也不设置返回的Dimension对象的width 和height值,那么仅凭测试是不可能检测到这类的错误。 .42OSV
^u74WN
bL%)k61G_v
`w}"0+V
不幸的是,随着时间的推移,客户代码可能会改变返回的Dimension对象的值,这个时候,追寻错误的根源是件枯燥且费时的事情,尤其是在多线程环境中。 c`Cn9bX
Ky|0IKE8Z
V-|}.kOH2
u,q#-d0g;
更好的方式是让getValues()返回拷贝: %Z|*!A+wN5
^~od*:
5{[0Clb)
UYUdIIoL
public synchronized Dimension getValues(){ 8a{g EZT,
return new Dimension (d.x, d.y); j.*}W4`Q_
} 3{=4q
UK/k?0
<Th.}=
R!ij CF\
现在,Example对象的内部状态就安全了。调用者可以根据需要改变它所得到的拷贝的状态,但是要修改Example对象的内部状态,必须通过setValues()才可以。 vbU{Et\^
O!jCQ{ T
Nt?B(.G
xOH@V4z:
三、常见错误3#:不必要的克隆 }^t?v*kcA
W:G*t4i
1^mO"nX
Ym6[~=~EK
我们现在知道了get方法应该返回内部数据对象的拷贝,而不是引用。但是,事情没有绝对: I nk76-
[LK
9^/V
fQdQ[
_'Z@ < ,L
/*** Example class.The value should never * be negative.*/ KAGq\7
public class Example{ O #"O.GX<
private Integer i = new Integer (0); 6IA~bkc}
public Example (){ } "#%T*c{Tf0
IN"qJ3<k
/*** Set x. x must be nonnegative* or an exception will be thrown*/ {fWZ n
public synchronized void setValues (int x) throws IllegalArgumentException{ fsu'W]f
if (x < 0) 4U3T..wA
throw new IllegalArgumentException(); ul
E\>5O4h
i = new Integer (x); {.AFg/Z
} rHznXME$wZ
xYbF76B
public synchronized Integer getValue(){ MWB?V?qPSC
// We can’t clone Integers so we makea copy this way. @U,cj>K
return new Integer (i.intValue()); e>/PW&Z8Z
} '`n\YO.N
} ]M.ufbg uq
+-ue={'
LYT0 XB)A
aX!J0&3
这段代码是安全的,但是就象在错误1#那样,又作了多余的工作。Integer对象,就象String对象那样,一旦被创建就是不可变的。因此,返回内部Integer对象,而不是它的拷贝,也是安全的。 $%/Zm*H
{az8*MR=X
I aW8
1cPi>?R:
方法getValue()应该被写为: )<+Z,6
[e (-
j&F&wRD%r
|yS4um(w
public synchronized Integer getValue(){ iC
2:P~
// ’i’ is immutable, so it is safe to return it instead of a copy. + IMP<
return i; U`D"L4},.
} xZ .:H&0G
6;ICX2Wq'
`*!.B
}a8N!g
Java程序比C++程序包含更多的不可变对象。JDK 所提供的若干不可变类包括: {oO!v}]
|(Sqd;#v
I;=}@]9
.S[5CO^
?Boolean u4%-e)$X
?Byte Q16RDQ*
?Character % 30&6 "
?Class Oa{M9d,l
?Double ~s+\Y/@A
?Float 9+pnpaZB0
?Integer dP )YPy_`
?Long X)\t=><<
?Short g8^ $,
?String Dp>/lkk.
?大部分的Exception的子类 VEJ Tw
(xMAo;s_
T<! TmG
Ta~Ei=d^
四、常见错误4# :自编代码来拷贝数组 LFtnSB8
t~e.LxN
*c. *e4uzF
(T,ST3{*k
Java允许你克隆数组,但是开发者通常会错误地编写如下的代码,问题在于如下的循环用三行做的事情,如果采用Object的clone方法用一行就可以完成: hX8;G!/
21W>}I"0?
GDhg
VOW(
PE-VxRN)
public class Example{ 5s8k^n"A
private int[] copy; ZfoI7<?33
/*** Save a copy of ’data’. ’data’ cannot be null.*/ ``U>9S"p)
public void saveCopy (int[] data){ %-> X$,Q
:
copy = new int[data.length]; {o AJL
for (int i = 0; i < copy.length; ++i) k]w;(<
copy = data; ZxY%x/K
} ^C_ ;uz
} o#CNr5/
Z+JPxe#7
EQ\/I(
=l
*$Bx#0J8
这段代码是正确的,但却不必要地复杂。saveCopy()的一个更好的实现是: y@e/G3
rdSkGb
>E6w,Ab
p{NVJ^!+
void saveCopy (int[] data){ m>DBO|`
try{ ]9F$/M#
copy = (int[])data.clone(); LS
<\%A}
}catch (CloneNotSupportedException e){ 6;Wns'
// Can’t get here. a/^ojn
} PVLLuv
} y38x^fuYJ~
lXjhT
"MOM@4\
,Y ./9F
如果你经常克隆数组,编写如下的一个工具方法会是个好主意: }}G`yfs}r
"w7wd5h
#-W
a3P
wF[%+n (*
static int[] cloneArray (int[] data){ 8`Ih>
Dc
try{ EIug)S~
return(int[])data.clone(); g:8k,1y5
}catch(CloneNotSupportedException e){ %OE
(?~dq
// Can’t get here. fAYp\k
} 3qggdi
} wNMf-~
rrz^LD
&!N9.e:-]
oR[-F+__
这样的话,我们的saveCopy看起来就更简洁了: M-7^\wXTA
F}MjZZj(U=
ecvQEK2L
.|UIZwW0
void saveCopy (int[] data){ ON [F
copy = cloneArray ( data); ;XagLy
} <Ukeq0
>W>3w
#7v=#Jco
U'*~Ju
五、常见错误5#:拷贝错误的数据 v-(dh5e`
H
GBSuTu8
`]F}O \H
grVPu! B;
有时候程序员知道必须返回一个拷贝,但是却不小心拷贝了错误的数据。由于仅仅做了部分的数据拷贝工作,下面的代码与程序员的意图有偏差: &+|bAn9AJ
b,hRk1
<~z@GMQCf
T\v~"pMu*0
import java.awt.Dimension; 7/e25LS!`U
/*** Example class. The height and width values should never * be 9w!PA-) L
negative. */ q8SHFKE
public class Example{ :6TLT-B
static final public int TOTAL_VALUES = 10; /`s{!t#Y
private Dimension[] d = new Dimension[TOTAL_VALUES]; %1\~OnT
public Example (){ } Mh"iyDGA
{c}n."`
/*** Set height and width. Both height and width must be nonnegative * or an exception will be thrown. */ br;~}GR_h
public synchronized void setValues (int index, int height, int width) throws IllegalArgumentException{ cB0"vbdO
if (height < 0 || width < 0) 3;?DKRIcX
throw new IllegalArgumentException(); z"\<GmvB
if (d[index] == null) <IBWA0A=8a
d[index] = new Dimension();
Lo*vt42{4
d[index].height = height; ech1{v\B|
d[index].width = width; v`&>m'
} \ lW*.<
public synchronized Dimension[] getValues() Lky T4HC8n
throws CloneNotSupportedException{ lk4U/:
return (Dimension[])d.clone(); o%E-K=a
} b
Bkg/p]
} ~P fk
>)t-Zh:n
{Q021*xt/
>@tJ7mM
这儿的问题在于getValues()方法仅仅克隆了数组,而没有克隆数组中包含的Dimension对象,因此,虽然调用者无法改变内部的数组使其元素指向不同的Dimension对象,但是调用者却可以改变内部的数组元素(也就是Dimension对象)的内容。方法getValues()的更好版本为: }6To(*
9]gV#uF
&ox5eX(
ffy,ds_7
public synchronized Dimension[] getValues() throws CloneNotSupportedException{ <YAs0
Dimension[] copy = (Dimension[])d.clone(); ,l#f6H7p
for (int i = 0; i < copy.length; ++i){ ]D_
AZI
// NOTE: Dimension isn’t cloneable. wvI}|c
if (d != null)
Iuve~ugO
copy = new Dimension (d.height, d.width); 04c`7[
} G&q@B`I
return copy; $Yj4&Two<
} PTpGZ2FZ
^S:I38gR#q
H<Zs2DP`
Va$JfWef
在克隆原子类型数据的多维数组的时候,也会犯类似的错误。原子类型包括int,float等。简单的克隆int型的一维数组是正确的,如下所示: RY}:&vWDk
YK V"bI
P\3H<?@4
Z8:'_#^@a[
public void store (int[] data) throws CloneNotSupportedException{ DI\^&F)3T2
this.data = (int[])data.clone(); ~>uu1[/
// OK _]M:
} 7
V=%&+
m }\L i]
^ux"<?
YR.f`-<Z
拷贝int型的二维数组更复杂些。Java没有int型的二维数组,因此一个int型的二维数组实际上是一个这样的一维数组:它的类型为int[]。简单的克隆int[][]型的数组会犯与上面例子中getValues()方法第一版本同样的错误,因此应该避免这么做。下面的例子演示了在克隆int型二维数组时错误的和正确的做法: YH3[Jvzf4
2 W Wr./q
#{~3bgY
72sBx3 ;
public void wrongStore (int[][] data) throws CloneNotSupportedException{ *40Z}1ng
this.data = (int[][])data.clone(); // Not OK! L{u1_
} $uUJV% EX
public void rightStore (int[][] data){ =UV=F/Af^
// OK! ^20x\K
this.data = (int[][])data.clone(); J1d|L|M
for (int i = 0; i < data.length; ++i){ *oru;=D@8
if (data != null) 9=G
dj!L
this.data = (int[])data.clone(); (h27SLYm
} JT*Pm"}
} An=Q`Uxt/
&\Yd)#B/
9S{?@*V
vW03nt86
'?b.t2
六、常见错误6#:检查new 操作的结果是否为null % ^&D,
{ud^+I&
(Ek=0;Cr
,CjJO -
Java编程新手有时候会检查new操作的结果是否为null。可能的检查代码为: Yd$64d7,h
!Nno@SP@
;aUI3n%
YbE1yOJ&m
Integer i = new Integer (400); 'q*:+|"
if (i == null) -1 _7z{.
throw new NullPointerException(); 9Jp"E5Ql)
}$1Aw%p^
0:k ~lz
ii]'XBSVd
检查当然没什么错误,但却不必要,if和throw这两行代码完全是浪费,他们的唯一功用是让整个程序更臃肿,运行更慢。 <>K@#|%Y&
nuX W/7M
\ /6m
:Qklbd[9qF
C/C++程序员在开始写java程序的时候常常会这么做,这是由于检查C中malloc()的返回结果是必要的,不这样做就可能产生错误。检查C++中new操作的结果可能是一个好的编程行为,这依赖于异常是否被使能(许多编译器允许异常被禁止,在这种情况下new操作失败就会返回null)。在java 中,new 操作不允许返回null,如果真的返回null,很可能是虚拟机崩溃了,这时候即便检查返回结果也无济于事。 oxL4* bqZ
5A*'@Fr'G
七、常见错误7#:用== 替代.equals Abf=b<bu
kC2_&L
在Java中,有两种方式检查两个数据是否相等:通过使用==操作符,或者使用所有对象都实现的.equals方法。原子类型(int, flosat, char 等)不是对象,因此他们只能使用==操作符,如下所示: 0-w^y<\
POvpaPAZ<
od~`q4p1(-
0!,)7
int x = 4; i4oBi]$T
int y = 5; /WIHG0D
if (x == y) 4C_-MJI
System.out.println ("Hi"); Y4dTv<=K@i
// This ’if’ test won’t compile. Zx}.mt#}8
if (x.equals (y)) IWcYa.=tZ
System.out.println ("Hi"); 7<VfE`Q3
Ig6>+Mw
yD!V;?EnK
5-pz/%,
对象更复杂些,==操作符检查两个引用是否指向同一个对象,而equals方法则实现更专门的相等性检查。 Ctxx.MM
GQ8r5V4:
_[W`!#"
y3!=0uPf
更显得混乱的是由java.lang.Object 所提供的缺省的equals方法的实现使用==来简单的判断被比较的两个对象是否为同一个。 _A.?:'-
zorTZ #5
'E,Bl]8C5
xbA% 'p
许多类覆盖了缺省的equals方法以便更有用些,比如String类,它的equals方法检查两个String对象是否包含同样的字符串,而Integer的equals方法检查所包含的int值是否相等。 ;{inhiySN
4GRmo"S
V
F'!
OPN
:{tvAdMl7
大部分时候,在检查两个对象是否相等的时候你应该使用equals方法,而对于原子类型的数据,你用该使用==操作符。 U;:,$]+
UI:{*N**Z
0|:Ic,
<`q|6XWL
八、常见错误8#: 混淆原子操作和非原子操作 3rZ" T
ft[g1
HYPFe|t/
VPTT*a`
Java保证读和写32位数或者更小的值是原子操作,也就是说可以在一步完成,因而不可能被打断,因此这样的读和写不需要同步。以下的代码是线程安全(thread safe)的: WLWE%bDP
[@= [<
_r
^q,KRut
)gCHwu
public class Example{ gH0B[w ]
private int value; // More code here... [#td
public void set (int x){ [lA[wCw
// NOTE: No synchronized keyword BifA&o%
this.value = x; +2w54X%?M
} zHB{I(q
} tGd<{nF% 2
N)h>Ie
7-ba-[t#A
B<[;rk
不过,这个保证仅限于读和写,下面的代码不是线程安全的: \sC0om,
-&kQlr
%`}CbD6
%MP s}B
public void increment (){ vO{ijHKE
// This is effectively two or three instructions: J6 [x(T
// 1) Read current setting of ’value’. G"TPu_g
// 2) Increment that setting. @\!wW-:A
// 3) Write the new setting back. 9rao&\eH
++this.value; #G.3a]p}"
} i?AZ|Ha[
\MtiLaI"
?GFxJ6!%I
$hSu~}g
在测试的时候,你可能不会捕获到这个错误。首先,测试与线程有关的错误是很难的,而且很耗时间。其次,在有些机器上,这些代码可能会被翻译成一条指令,因此工作正常,只有当在其它的虚拟机上测试的时候这个错误才可能显现。因此最好在开始的时候就正确地同步代码:
OV8b~k4=
31>k3IP&
-t6d`p;dR
m4Wn$Z
public synchronized void increment (){ h0.Fstf]
++this.value; qqzQKN
} R|+R4'
v[a#>!;s
EJWMr`zdn
`FJnR~d
九、常见错误9#:在catch 块中作清除工作 &{]%=stI
Xp} vJl
-Ez|
HnP;1Gi
一段在catch块中作清除工作的代码如下所示: &vMH
AZd
GmL |7 6
(C-z8R
Z6
{my=Li<