代码审查是消灭Bug最重要的方法之一,这些审查在大多数时候都特别奏效。由于代码审查本身所针对的对象,就是俯瞰整个代码在测试过程中的问题和Bug。并且,代码审查对消除一些特别细节的错误大有裨益,尤其是那些能够容易在阅读代码的时候发现的错误,这些错误往往不容易通过机器上的测试识别出来。本文就常见的Java代码中容易出现的问题提出一些建设性建议,以便您在审查代码的过程中注意到这些常见的细节性错误。 i'z(`"
yUvn h
.Ix[&+LsY
通常给别人的工作挑错要比找自己的错容易些。别样视角的存在也解释了为什么作者需要编辑,而运动员需要教练的原因。不仅不应当拒绝别人的批评,我们应该欢迎别人来发现并指出我们的编程工作中的不足之处,我们会受益匪浅的。 %18%T{|$e
G=&nwSL
}r|$\ms
^=y%s
正规的代码审查(code inspection)是提高代码质量的最强大的技术之一,代码审查?由同事们寻找代码中的错误?所发现的错误与在测试中所发现的错误不同,因此两者的关系是互补的,而非竞争的。 y>] Yq-
'6GW.;
lMe+.P|
=gHUY&sPu8
如果审查者能够有意识地寻找特定的错误,而不是靠漫无目的的浏览代码来发现错误,那么代码审查的效果会事半功倍。在这篇文章中,我列出了11个Java编程中常见的错误。你可以把这些错误添加到你的代码审查的检查列表(checklist)中,这样在经过代码审查后,你可以确信你的代码中不再存在这类错误了。 ;Mz7emt
?D=C8[NEX
98lz2d/Fcq
rVB\\
一、常见错误1# :多次拷贝字符串 a&<_M$J&
FBS]U$1
cg^=F_h
Uwg*kJ3H
测试所不能发现的一个错误是生成不可变(immutable)对象的多份拷贝。不可变对象是不可改变的,因此不需要拷贝它。最常用的不可变对象是String。 P9q ZjBS
w-LaSJ(T
yw*|
H T
0+Q;a
如果你必须改变一个String对象的内容,你应该使用StringBuffer。下面的代码会正常工作: j{Sbf04
*@g>~q{`
a6 w'.]m
0D&-BAzi
String s = new String ("Text here"); b&*N
a'?V:3 ]
n= FOB0=
oq2-)F2/
但是,这段代码性能差,而且没有必要这么复杂。你还可以用以下的方式来重写上面的代码: @# GS4I
z~#d@c\
bd)Sb?
&+ UnPE(
String temp = "Text here";
#M|q}jA|
String s = new String (temp); U$LI~XZM
/]9(InM9/
KU;J2Kt
8JU{]Z!G<;
但是这段代码包含额外的String,并非完全必要。更好的代码为: [I78<IJc
="
pNE#
WMnxN34
op61-:q/
String s = "Text here"; _Q7]Dw/w\
/g$8JL
QI`&N(n
+<fT\Oq#
二、常见错误2#: 没有克隆(clone)返回的对象 @s|yH"
fwv.^kx
Xr{
r&Rl
{~ 1
~V
封装(encapsulation)是面向对象编程的重要概念。不幸的是,Java为不小心打破封装提供了方便??Java允许返回私有数据的引用(reference)。下面的代码揭示了这一点: \ \g Aa-}:
i&0Zli
l f_q6y
R{q<V uN
import java.awt.Dimension; yZ,S$tSR
/***Example class.The x and y values should never*be negative.*/ G|Du/XYh
public class Example{ BCE}Er&
private Dimension d = new Dimension (0, 0); v.08,P{b
public Example (){ } 8TK&i,
`dH[&=S
/*** Set height and width. Both height and width must be nonnegative * or an exception is thrown.*/ y]
Io`w(>
public synchronized void setValues (int height,int width) throws IllegalArgumentException{ 5z8!Nmb/
if (height < 0 || width < 0) }*2q7K2bj
throw new IllegalArgumentException(); Yy4?|wVl
d.height = height; yW&|ZJF?
d.width = width; ~*|0yPFg
} Ip/_uDi+!Z
2d,q?VH$
public synchronized Dimension getValues(){ edt(Zzk@3-
// Ooops! Breaks encapsulation D&0@k'
return d; 1D42+cy
} ;:nO5VFOg
} TSQ/{=r
kKV`9&dZe
f^8,Z+n
o2cZ
Example类保证了它所存储的height和width值永远非负数,试图使用setValues()方法来设置负值会触发异常。不幸的是,由于getValues()返回d的引用,而不是d的拷贝,你可以编写如下的破坏性代码: Rc2| o.'y
9/rX%
)q?z"F|
D2$"!7O1H
Example ex = new Example(); LQ5 WS
Dimension d = ex.getValues(); \Uun2.K
d.height = -5; VXforI
d.width = -10; RW!D!~
.d:sQ\k~=
k$nQY
Ic4>kKh
现在,Example对象拥有负值了!如果getValues() 的调用者永远也不设置返回的Dimension对象的width 和height值,那么仅凭测试是不可能检测到这类的错误。 T"Ph@I<
kd9rvy0oK
N%{&%C 6{
[nBlHI;&
不幸的是,随着时间的推移,客户代码可能会改变返回的Dimension对象的值,这个时候,追寻错误的根源是件枯燥且费时的事情,尤其是在多线程环境中。 cCN[c)[c|
z5Hz-.
rkW*C'2fz
g]O"l?xx1D
更好的方式是让getValues()返回拷贝: S5]rIcM
4;8
Z?.
/m%Y.:g
X.V7od>
public synchronized Dimension getValues(){ Y(
n# =
return new Dimension (d.x, d.y); -sl]
funRy
} *KSQ^.sYh
A?'Tigi
fQq'_q5
uqPagt<
现在,Example对象的内部状态就安全了。调用者可以根据需要改变它所得到的拷贝的状态,但是要修改Example对象的内部状态,必须通过setValues()才可以。 w$4fS
smW
7zGE
_v:t$k#sN
` WIv|S
三、常见错误3#:不必要的克隆 H->J.5~,K
`)$'1,]u
0L1NZY^!
Kf#9-.}?
我们现在知道了get方法应该返回内部数据对象的拷贝,而不是引用。但是,事情没有绝对:
`b 6j7
WrBiAh,
hOjy$Z
uQgv ;jsPz
/*** Example class.The value should never * be negative.*/ !2>MaV1,
public class Example{ 'B0=
"7
private Integer i = new Integer (0); FYh+G-Y#
public Example (){ } swEE >=
}cuU5WQ?%
/*** Set x. x must be nonnegative* or an exception will be thrown*/ 7jP
C{W
public synchronized void setValues (int x) throws IllegalArgumentException{ >DqV^%2l
if (x < 0) W,'30:#Fr7
throw new IllegalArgumentException(); V*HkFT
i = new Integer (x); qBL>C\V +
} I[u%kir
{1a%CsCM
public synchronized Integer getValue(){ q))rlMo
// We can’t clone Integers so we makea copy this way. J eCKnt=
return new Integer (i.intValue()); eP6`"<UM
} 4 0as7.q
} \!D <u'n
@AM;58.
t;g=@o9YA
-'ff0l
这段代码是安全的,但是就象在错误1#那样,又作了多余的工作。Integer对象,就象String对象那样,一旦被创建就是不可变的。因此,返回内部Integer对象,而不是它的拷贝,也是安全的。 ~wtl\-cY
Qf0 ]7
;VVKn=X=S=
3sUTdCnNf
方法getValue()应该被写为: <7-Qn(m,
c8'a<<sj
`<}Q4p
BllS3I}V
public synchronized Integer getValue(){ l\+^.ezD
// ’i’ is immutable, so it is safe to return it instead of a copy. Ll4/P[7:?
return i; QnHb*4<
} <0R$yB
OUo N
l-?B1gd,l
K ?R*
)_
Java程序比C++程序包含更多的不可变对象。JDK 所提供的若干不可变类包括: t]dtBt].:
@D$^-
S6
OFc Lh
5W%^g_I
?Boolean W|C>X=zTi
?Byte bA/,{R
?Character :
T` Ni
?Class DcjF$E
?Double Tf0"9
?Float j.}@ 9
?Integer \L6kCY
?Long "#<P--E 9
?Short %8U/!(.g
?String aHhr_.>X
?大部分的Exception的子类 g3fxf(iY(
?o0ro?9j
y~16o
U"SH
fI:
四、常见错误4# :自编代码来拷贝数组 3vcKK;qCB
*!mT#Vm^
)iKV"jsC
*F4"mr|\
Java允许你克隆数组,但是开发者通常会错误地编写如下的代码,问题在于如下的循环用三行做的事情,如果采用Object的clone方法用一行就可以完成: "xr=:[n[
VXfp=JE
s2IjZF {
u*W6fg/"
public class Example{ 7,^.h<@K
private int[] copy; hN#A3FFo L
/*** Save a copy of ’data’. ’data’ cannot be null.*/ [5"F=tT7WP
public void saveCopy (int[] data){ w6Owfq'v
copy = new int[data.length]; #8{U0 7]"
for (int i = 0; i < copy.length; ++i) YA?46[:
copy = data; SJ
ay
} F4b$
} ^yl)c
\`
5LYzX+a)
Kw/7X[|'G
o@5zf{-
这段代码是正确的,但却不必要地复杂。saveCopy()的一个更好的实现是: CogN1,GJ
]cKxYX)J
Rr% CP[bH
ru>c\X^|
void saveCopy (int[] data){ y\C_HCU H
try{ L-,C5^
copy = (int[])data.clone(); '0jjoZ:
}catch (CloneNotSupportedException e){ (X{o =co,
// Can’t get here. 'xG:v)(
} F\Z|JCA
} \LEUreTn
mne?r3d
9&O7