社区应用 最新帖子 精华区 社区服务 会员列表 统计排行 社区论坛任务 迷你宠物
  • 3381阅读
  • 0回复

审查Java代码的十一种常见错误

级别: 终身会员
发帖
3743
铜板
8
人品值
493
贡献值
9
交易币
0
好评度
3746
信誉值
0
金币
0
所在楼道
代码审查是消灭Bug最重要的方法之一,这些审查在大多数时候都特别奏效。由于代码审查本身所针对的对象,就是俯瞰整个代码在测试过程中的问题和Bug。并且,代码审查对消除一些特别细节的错误大有裨益,尤其是那些能够容易在阅读代码的时候发现的错误,这些错误往往不容易通过机器上的测试识别出来。本文就常见的Java代码中容易出现的问题提出一些建设性建议,以便您在审查代码的过程中注意到这些常见的细节性错误。 (xjqB{U  
4;",@}  
/ O|Td'Z  
  通常给别人的工作挑错要比找自己的错容易些。别样视角的存在也解释了为什么作者需要编辑,而运动员需要教练的原因。不仅不应当拒绝别人的批评,我们应该欢迎别人来发现并指出我们的编程工作中的不足之处,我们会受益匪浅的。 k q/t]%(  
6zELe.tq  
b "`ru~]  
{_?T:`  
 正规的代码审查(code inspection)是提高代码质量的最强大的技术之一,代码审查?由同事们寻找代码中的错误?所发现的错误与在测试中所发现的错误不同,因此两者的关系是互补的,而非竞争的。 qAnA=/k`  
5F% h>tqh  
jM{(8aUG  
t00\yb^vJ8  
  如果审查者能够有意识地寻找特定的错误,而不是靠漫无目的的浏览代码来发现错误,那么代码审查的效果会事半功倍。在这篇文章中,我列出了11个Java编程中常见的错误。你可以把这些错误添加到你的代码审查的检查列表(checklist)中,这样在经过代码审查后,你可以确信你的代码中不再存在这类错误了。 |C&%S"*+D  
@Pd) %'s  
BYkVg2D(  
8  /5sv  
  一、常见错误1# :多次拷贝字符串 #_?426Wfs  
H^]Nmd8Q)  
ce 7Yr*ZB  
L?AM&w-cg9  
  测试所不能发现的一个错误是生成不可变(immutable)对象的多份拷贝。不可变对象是不可改变的,因此不需要拷贝它。最常用的不可变对象是String。 -ryDsq  
"``W6W-(  
^uKnP>*l  
A%.J%[MVz  
  如果你必须改变一个String对象的内容,你应该使用StringBuffer。下面的代码会正常工作: Q:'qw#P/C  
'Wo?%n  
ocb%&m ;i  
|) x'  
String s = new String ("Text here"); 6~:W(E}  
/5L'9e  
UIC\CP d  
wUh3Hd'  
  但是,这段代码性能差,而且没有必要这么复杂。你还可以用以下的方式来重写上面的代码: -lJx%9>  
x*5 Ch~<k  
D!l [3  
wrZ7Sr!/V  
String temp = "Text here"; UrD=|-r`  
String s = new String (temp);  ;Puy A  
8@9hU`H8l  
6R$ F =MB  
9~LpO>-  
  但是这段代码包含额外的String,并非完全必要。更好的代码为: g&oc=f`  
mf Wz@=0  
8MYLXW6  
zgEr,nF  
String s = "Text here"; vkDZv@  
GoGohsj  
<M5{.`o  
~`nm<   
  二、常见错误2#: 没有克隆(clone)返回的对象 =;'ope(?S  
F[o+p|nF  
,yB?~  
"ZA$"^  
  封装(encapsulation)是面向对象编程的重要概念。不幸的是,Java为不小心打破封装提供了方便??Java允许返回私有数据的引用(reference)。下面的代码揭示了这一点: 4?P%M"\Iv  
Fi?U)T+%+  
i?1js! 8  
qK 9L+i  
import java.awt.Dimension; kxr6sO~  
/***Example class.The x and y values should never*be negative.*/ :,xyVb+  
public class Example{ ^P3g9'WK  
  private Dimension d = new Dimension (0, 0); }a #b$]Y  
  public Example (){ } .!7Fe)(x  
;PP_3`  
  /*** Set height and width. Both height and width must be nonnegative * or an exception is thrown.*/ X]3l| D  
  public synchronized void setValues (int height,int width) throws IllegalArgumentException{ =hZ&66  
   if (height < 0 || width < 0) P;HVLflu  
    throw new IllegalArgumentException(); al3BWRq'f  
    d.height = height; \Y>#^b?  
     d.width = width; )V9Mcr*Ce6  
  } L V[66<T  
4U LJtM3  
  public synchronized Dimension getValues(){ K4h-4Qbn  
   // Ooops! Breaks encapsulation SG(%d^x`R  
   return d; fY)4]=L  
  } a6 #{2q  
} #G[ *2h~99  
s&_IWala  
(d5vH)+ A  
N>cp>&jV  
  Example类保证了它所存储的height和width值永远非负数,试图使用setValues()方法来设置负值会触发异常。不幸的是,由于getValues()返回d的引用,而不是d的拷贝,你可以编写如下的破坏性代码: -6em*$k^  
X d19GP!  
n!CP_  
: e0R7sj  
Example ex = new Example(); ]sm0E@1  
Dimension d = ex.getValues(); Y7b,td1  
d.height = -5; cW~6@&zp  
d.width = -10; ]$?zT`>(F  
( TbB?X}  
||*&g2Y  
UL@5*uiX  
  现在,Example对象拥有负值了!如果getValues() 的调用者永远也不设置返回的Dimension对象的width 和height值,那么仅凭测试是不可能检测到这类的错误。 L_.xr ?  
R.T?ZF  
ki*79d"$  
QvK]<HEr  
  不幸的是,随着时间的推移,客户代码可能会改变返回的Dimension对象的值,这个时候,追寻错误的根源是件枯燥且费时的事情,尤其是在多线程环境中。 DS[l,x  
x]%4M\T``  
,,wyydG  
D@lAT#vA  
  更好的方式是让getValues()返回拷贝: y ? {PoNI  
]'1N_m]?  
69<rsp(p  
9>.<+b(>!'  
public synchronized Dimension getValues(){ ,,C~j`F  
return new Dimension (d.x, d.y);  ycAi(K  
} @6I[{{>X  
Jq?^8y  
2'O!~8U  
yaYIgG  
  现在,Example对象的内部状态就安全了。调用者可以根据需要改变它所得到的拷贝的状态,但是要修改Example对象的内部状态,必须通过setValues()才可以。 6%tiB?  
"z(fBnv  
c@ZkX]g  
0=(-8vwd  
  三、常见错误3#:不必要的克隆 WO \lny!  
gn e #v  
yw3U"/yw  
$V{- @=  
  我们现在知道了get方法应该返回内部数据对象的拷贝,而不是引用。但是,事情没有绝对: T0np<l]A  
w'!}(Z5X?  
U7f&N  
(Aov}I+  
/*** Example class.The value should never * be negative.*/ ;t@ 3Go  
public class Example{ %;B(_ht<-w  
  private Integer i = new Integer (0); vCU&yXGl  
  public Example (){ } 1 [~|  
x1hs19s  
  /*** Set x. x must be nonnegative* or an exception will be thrown*/ JG+g88  
  public synchronized void setValues (int x) throws IllegalArgumentException{ Z+"E*  
   if (x < 0) "|l oSf@  
    throw new IllegalArgumentException(); ).O2_<&?F  
    i = new Integer (x); wJ]$'c3  
  } ezq q@t9  
N:gstp  
  public synchronized Integer getValue(){ )/N Xh'  
   // We can’t clone Integers so we makea copy this way. xdTzG4  
   return new Integer (i.intValue()); M'!!EQo  
  } hc p'+:  
} ,n,7.m.D  
pS|JDMo  
m(7_ZiL=  
V@+<,tjq  
  这段代码是安全的,但是就象在错误1#那样,又作了多余的工作。Integer对象,就象String对象那样,一旦被创建就是不可变的。因此,返回内部Integer对象,而不是它的拷贝,也是安全的。 dv4r\ R^  
(m =u;L"o  
ow>[#.ua  
/+JP~ K  
  方法getValue()应该被写为: Zkb,v!l  
4S{l>/I  
)V+Dqh,-g  
"*>QxA%c4  
public synchronized Integer getValue(){ GF.g'wYc)Y  
// ’i’ is immutable, so it is safe to return it instead of a copy. 0wE8Gm G  
return i; cdU >iB,  
} N(({2'Rr  
r{:la56Xd  
I}Gl*@K&O  
Om?:X!l"  
  Java程序比C++程序包含更多的不可变对象。JDK 所提供的若干不可变类包括: cX=b q_  
Dil4ut- $  
HjF'~n  
[Xo J7  
  ?Boolean '?!<I  
   ?Byte &MGgO\|6  
   ?Character Z`1o#yZ  
   ?Class ])ZJ1QL1  
   ?Double h|/*yTuN.y  
   ?Float VT~ ^:-]  
   ?Integer qI%9MI;BV  
   ?Long QX~72X=(  
   ?Short xyJgHbml  
   ?String <wGT s6  
   ?大部分的Exception的子类 #( Yb lY  
qP.VK?jF|  
o _,$`nEJ  
r Xk   
 四、常见错误4# :自编代码来拷贝数组 : w`i  
6V_5BpXt  
Pc:'>,3!V3  
~(doy@0M  
  Java允许你克隆数组,但是开发者通常会错误地编写如下的代码,问题在于如下的循环用三行做的事情,如果采用Object的clone方法用一行就可以完成: FU v)<rK  
$YO]IK$  
N|# x9mE  
V9 t:JY  
public class Example{ GH)+yD[o  
  private int[] copy; ~|d?o5W  
  /*** Save a copy of ’data’. ’data’ cannot be null.*/ %KVRiX  
  public void saveCopy (int[] data){ 5>k~yaju/  
   copy = new int[data.length]; |c+N)F B  
   for (int i = 0; i < copy.length; ++i) P6Z,ci17  
    copy = data; JC?V].) y5  
  } W;x LuKIG  
} kd2'-9  
@P*P8v8:  
).#D:eO[~  
%;XuA*e  
  这段代码是正确的,但却不必要地复杂。saveCopy()的一个更好的实现是: $,@ +Ua  
=|t1eSzc  
JU`'?b  
)t 7HioQ  
void saveCopy (int[] data){ I Y-5/  
  try{ :95_W/l  
   copy = (int[])data.clone(); -8J@r2\  
  }catch (CloneNotSupportedException e){ mp$II?hZ*  
   // Can’t get here. \Hx#p`B%  
  } u6&Ixi/s'  
} YO4ppL~xe  
f2K3*}P  
$fpDABf  
'`VO@a  
  如果你经常克隆数组,编写如下的一个工具方法会是个好主意: ;iI2K/ 3  
s5|)4Z ac  
8{^GC(W{]  
Yy;1N{dbT  
static int[] cloneArray (int[] data){ Z`h_oK#y15  
  try{ \}&w/.T  
   return(int[])data.clone(); dufHd  
  }catch(CloneNotSupportedException e){ F,$$N>  
   // Can’t get here. AyXKhj#Ml  
  } F>{uB!!L4  
} BP><G^  
y,eoTmaI  
{*  _ W  
uPD_s[  
  这样的话,我们的saveCopy看起来就更简洁了: <}>-ip?  
-P uVI5L<  
Ho{?m^  
lt2& uYgp  
void saveCopy (int[] data){ j[ YTg]  
  copy = cloneArray ( data); Ppn ZlGQ6  
} E)SOcM)  
d`*vJ#$> 2  
+K4v"7C V  
^HKaNk<  
  五、常见错误5#:拷贝错误的数据 <1K: G/!  
ol>=tk 8}  
 C3Z(k}  
{-Oc8XI/  
  有时候程序员知道必须返回一个拷贝,但是却不小心拷贝了错误的数据。由于仅仅做了部分的数据拷贝工作,下面的代码与程序员的意图有偏差: u"3cSuqy  
lw lW.C  
D#(Pg  
}=R|iz*,!  
import java.awt.Dimension; vx,6::%]  
/*** Example class. The height and width values should never * be )CU(~s|s  
negative. */ Gs?sO?j  
public class Example{ Xc<9[@  
  static final public int TOTAL_VALUES = 10; Cf 8 - %  
  private Dimension[] d = new Dimension[TOTAL_VALUES]; {i?K~| h  
  public Example (){ } a.Vs >1  
#rC+13  
  /*** Set height and width. Both height and width must be nonnegative * or an exception will be thrown. */ P=i |{vv(  
  public synchronized void setValues (int index, int height, int width) throws IllegalArgumentException{ :~(^b;yhZ  
   if (height < 0 || width < 0) ZACn_gd[5  
    throw new IllegalArgumentException(); C!A_PQ2y  
    if (d[index] == null) 6!V* :.(  
     d[index] = new Dimension(); Hh/#pGf2  
     d[index].height = height; SQRz8,sqkw  
     d[index].width = width; ~#xRoBy3  
  } RozsRt;i  
  public synchronized Dimension[] getValues() !T1i_  
   throws CloneNotSupportedException{ $ :P~21,  
    return (Dimension[])d.clone(); ZuON@(  
  } QpZhxp  
} P,], N)  
/FXfu  
&Vm[5XW  
e6/} M3B  
  这儿的问题在于getValues()方法仅仅克隆了数组,而没有克隆数组中包含的Dimension对象,因此,虽然调用者无法改变内部的数组使其元素指向不同的Dimension对象,但是调用者却可以改变内部的数组元素(也就是Dimension对象)的内容。方法getValues()的更好版本为: 3<SC`6'?  
m)2U-3*iX  
G]4+ Qr?  
]5L3[A4Vu  
public synchronized Dimension[] getValues() throws CloneNotSupportedException{ ;#Nci%<J\  
  Dimension[] copy = (Dimension[])d.clone(); {uG_)GFr0  
  for (int i = 0; i < copy.length; ++i){ 7~f l4*  
   // NOTE: Dimension isn’t cloneable. A).AAr  
   if (d != null) >l[N]CQ  
    copy = new Dimension (d.height, d.width); rGO 3  
  }  vpMv  
  return copy; au v\fR :  
} q3:' 69  
9dv~WtH>5  
247>+:7z  
M>#S z  
  在克隆原子类型数据的多维数组的时候,也会犯类似的错误。原子类型包括int,float等。简单的克隆int型的一维数组是正确的,如下所示: L*38T\  
IT"jtV  
 EZFWxR/  
\/G Y0s  
public void store (int[] data) throws CloneNotSupportedException{ ld6@&34  
  this.data = (int[])data.clone(); EORAx  
  // OK 8t"DQ Y-R  
} WNi<|A#T{  
 #pK)  
j_g9RmZT  
F3'G9Xf8Q=  
  拷贝int型的二维数组更复杂些。Java没有int型的二维数组,因此一个int型的二维数组实际上是一个这样的一维数组:它的类型为int[]。简单的克隆int[][]型的数组会犯与上面例子中getValues()方法第一版本同样的错误,因此应该避免这么做。下面的例子演示了在克隆int型二维数组时错误的和正确的做法: (x!bZ,fu  
1gr jK.x  
w[_Uv4M  
_69\#YvCG  
public void wrongStore (int[][] data) throws CloneNotSupportedException{ ' ga2C\)  
  this.data = (int[][])data.clone(); // Not OK! 5sUnEHN  
} YG|T;/-  
public void rightStore (int[][] data){ }Z=Qy;zk  
  // OK! pq`MO .R  
  this.data = (int[][])data.clone(); oPV"JGa/B4  
  for (int i = 0; i < data.length; ++i){ .:/@<V+K  
   if (data != null) `z.#O\@o  
    this.data = (int[])data.clone(); ]QQ"7_+  
  } ^m9cEl^:nQ  
} 4 n( f/  
W525:h52{  
T@XiG:b7  
4#uoPkLK  
o%iTYR :x  
   六、常见错误6#:检查new 操作的结果是否为null G[ea@u$?  
/cn_|DwN5  
UYUd IIoL  
|@F<ajlV  
  Java编程新手有时候会检查new操作的结果是否为null。可能的检查代码为: Y_B( R  
5 Da( DA  
[d}1Cq=_  
r+crE %-  
Integer i = new Integer (400); #wfR$Cd  
if (i == null) ;'kH<Iq  
throw new NullPointerException(); d0d2QRX  
YVi]f2F%  
AnQRSB (  
#e[5O| V~  
  检查当然没什么错误,但却不必要,if和throw这两行代码完全是浪费,他们的唯一功用是让整个程序更臃肿,运行更慢。 i\b2P2 `B  
:csLZqn[  
a6C ~!{'nW  
BVDo5^&W  
  C/C++程序员在开始写java程序的时候常常会这么做,这是由于检查C中malloc()的返回结果是必要的,不这样做就可能产生错误。检查C++中new操作的结果可能是一个好的编程行为,这依赖于异常是否被使能(许多编译器允许异常被禁止,在这种情况下new操作失败就会返回null)。在java 中,new 操作不允许返回null,如果真的返回null,很可能是虚拟机崩溃了,这时候即便检查返回结果也无济于事。 <T>f@Dn,  
WqO* vK!t  
 七、常见错误7#:用== 替代.equals jw)t"S/E  
>?tpGEZ\  
  在Java中,有两种方式检查两个数据是否相等:通过使用==操作符,或者使用所有对象都实现的.equals方法。原子类型(int, flosat, char 等)不是对象,因此他们只能使用==操作符,如下所示: inPGWG K]  
v>6r|{  
XR(kR{yo  
t1S\M%?  
int x = 4; SV >EB;<  
int y = 5; n@f@-d$m\<  
if (x == y) RY&~{yl$"1  
  System.out.println ("Hi"); 5{UGSz 1  
// This ’if’ test won’t compile. ,==lgM2V>  
if (x.equals (y)) M*Xzr .6  
  System.out.println ("Hi"); BH^q.p_#>X  
V Puzu|  
\} 5\^&}_  
W=drp>Uj  
  对象更复杂些,==操作符检查两个引用是否指向同一个对象,而equals方法则实现更专门的相等性检查。 VlH9ap  
MLl:)W*  
pmZr<xs   
4U3T..wA  
  更显得混乱的是由java.lang.Object 所提供的缺省的equals方法的实现使用==来简单的判断被比较的两个对象是否为同一个。 d?JVB  
1x]G/I*  
{ .AFg/Z  
ygHNAQG~  
  许多类覆盖了缺省的equals方法以便更有用些,比如String类,它的equals方法检查两个String对象是否包含同样的字符串,而Integer的equals方法检查所包含的int值是否相等。 &f$jpIyVX  
!#QD;,SE+  
:Fh* 4 &Z  
}0 Z3Lrv  
  大部分时候,在检查两个对象是否相等的时候你应该使用equals方法,而对于原子类型的数据,你用该使用==操作符。 ugz1R+f_4{  
vhKD_}}aP  
2B|3`trY4x  
 ^(y4]yZ  
  八、常见错误8#: 混淆原子操作和非原子操作 ufmFeeg  
lxbZM9A2  
q;+qIV&.:  
Z=ho7i  
  Java保证读和写32位数或者更小的值是原子操作,也就是说可以在一步完成,因而不可能被打断,因此这样的读和写不需要同步。以下的代码是线程安全(thread safe)的: |dvcDx0|K  
sy~mcH:%+  
oPi)#|jcb  
Ty>`r n  
public class Example{ Wjp<(aY[  
  private int value; // More code here... {az8*MR=X  
  public void set (int x){ n']@Spm  
   // NOTE: No synchronized keyword ,+XQ!y%  
   this.value = x; vjWS35i  
  } XS>4efCJ  
} J?{uG8)  
) E5ax~  
Xa36O5$4]9  
8I*fPf  
  不过,这个保证仅限于读和写,下面的代码不是线程安全的: |m~|  
v+Hu=RZE  
r*$KF!-dg  
%gN8-~$ 1  
public void increment (){ mR@iGl\\  
  // This is effectively two or three instructions: -k'=s{iy  
  // 1) Read current setting of ’value’. 6;ICX2Wq'  
  // 2) Increment that setting. ZC05^  
  // 3) Write the new setting back. o9JJ_-O"  
  ++this.value; }a8N!g  
} r3|vu"Uei  
~{xY{qL  
^#;2 Pd>  
>t.I,Zn  
  在测试的时候,你可能不会捕获到这个错误。首先,测试与线程有关的错误是很难的,而且很耗时间。其次,在有些机器上,这些代码可能会被翻译成一条指令,因此工作正常,只有当在其它的虚拟机上测试的时候这个错误才可能显现。因此最好在开始的时候就正确地同步代码: x\)-4w<P  
kj>XKZL10  
4o'0lz]  
p< XjiRq  
public synchronized void increment (){ OA[w|Tt  
  ++this.value; .iw+ #  
} :[F w c  
{R(q7ALR  
o+&/ N-t  
6x_8m^+m  
  九、常见错误9#:在catch 块中作清除工作 F<o J  
_T H'v:C  
o)w'w34FCT  
{jbOcx$t  
  一段在catch块中作清除工作的代码如下所示: =VDN9-/.  
pDW .Pav  
VF;%Z  
=>&d[G[m!  
OutputStream os = null; j  $L  
try{ %h^; "|Z  
  os = new OutputStream (); ugOcK Gf  
  // Do something with os here. a93Aj  
  os.close(); (g5T2(_6L  
}catch (Exception e){ 6ZX{K1_q  
  if (os != null) d^4!=^HN  
  os.close(); V;9.7v  
} 23 3jT@Z  
uV{cvq$jy  
y/E%W/3  
q^EG'\<^  
  尽管这段代码在几个方面都是有问题的,但是在测试中很容易漏掉这个错误。下面列出了这段代码所存在的三个问题: /1Ndir^c  
RxcX\:  
s(-$|f+s  
x-cg df  
  1.语句os.close()在两处出现,多此一举,而且会带来维护方面的麻烦。 L_O m<LO2  
TEv3;Z*N  
lRn>/7sg$  
b16\2%Ea1  
  2.上面的代码仅仅处理了Exception,而没有涉及到Error。但是当try块运行出现了Error,流也应该被关闭。 zK?[6n89f  
kz]qk15w  
pLNv\M+  
FK>8(M/  
  3.close()可能会抛出异常。 TtlZum\  
7h0LR7  
[8![UcMq  
8KN0z<  
  上面代码的一个更优版本为: ^C_ ;uz  
V4iN2  
0jG8Gmh!  
bDRl}^aO6  
OutputStream os = null; "RiY#=}sm  
try{ Z sv(/>  
  os = new OutputStream (); *}Vg]3$4  
  // Do something with os here. IOomBy:  
}finally{ wm_xH_{F  
  if (os != null) Dhv ^}m@  
   os.close(); 5V8WSnO  
} >E6w,Ab  
vT)FLhH6*  
 K<6)SL4  
0.qnbDw_  
  这个版本消除了上面所提到的两个问题:代码不再重复,Error也可以被正确处理了。但是没有好的方法来处理第三个问题,也许最好的方法是把close()语句单独放在一个try/catch块中。 [s"xOP9R  
AfB,`l`k  
s&TPG0W  
RX\%R  
  十、常见错误10#: 增加不必要的catch 块 Igrr"NuDZ  
2XNO*zbve  
a/ ^ojn  
3P N<J  
  一些开发者听到try/catch块这个名字后,就会想当然的以为所有的try块必须要有与之匹配的catch块。 %xPJJ $P  
7\HjQ7__  
y38x^fuYJ~  
&C6Z-bS"  
  C++程序员尤其是会这样想,因为在C++中不存在finally块的概念,而且try块存在的唯一理由只不过是为了与catch块相配对。 yp9vgUs  
!E#FzY!}Pl  
!W^P|:Qt  
~x4]^XS  
  增加不必要的catch块的代码就象下面的样子,捕获到的异常又立即被抛出: 5LMAy"  
bdbTK8-  
t}w<xe  
b9X"p*'p  
try{ a'r8J~:jy  
  // Nifty code here usc"m huQ  
}catch(Exception e){ n|q $=jE  
  throw e; clyZD`*  
}finally{ v)1@Ew=Y%  
  // Cleanup code here ;auT!a~a#  
} fAYp\ k  
wkc)2z   
}xJ ).D  
Y#7sDd!N|  
  不必要的catch块被删除后,上面的代码就缩短为: =jz [}5  
)jm!bR`  
N.(wR  
b v5BV  
try{ 4z6kFQgu  
  // Nifty code here |q!O~<H@  
}finally{ QN)EPS:y  
  // Cleanup code here *QH~ z2:[  
} xU9T8Lw  
5d|hP4fEc  
fkk&pu  
1K\z amBg  
  常见错误11#;没有正确实现equals,hashCode,或者clone 等方法 upi\pXv  
DXyRNE<G[C  
XN|[8+#U<@  
DsX+/)d  
  方法equals,hashCode,和clone 由java.lang.Object提供的缺省实现是正确的。不幸地是,这些缺省实现在大部分时候毫无用处,因此许多类覆盖其中的若干个方法以提供更有用的功能。但是,问题又来了,当继承一个覆盖了若干个这些方法的父类的时候,子类通常也需要覆盖这些方法。在进行代码审查时,应该确保如果父类实现了equals,hashCode,或者clone等方法,那么子类也必须正确。正确的实现equals,hashCode,和clone需要一些技巧。 JP{Y Q:NF  
ZW>iq M^9  
~'lYQ[7  
8GlRO4yd  
  小结 VRE[ vM'  
;2N: =Rv  
mM(Z8PA 9-  
uSQRI9/ir2  
  我在代码审查的时候至少遇到过一次这些错误,我自己也犯过其中的几个错误。好消息是只要你知道你在找什么错误,那么代码审查就很容易管理,错误也很容易被发现和修改。即便你找不到时间来进行正规的代码审查,以自审的方式把这些错误从你的代码中根除会大大节省你的调试时间。花时间在代码审查上是值得的。 @;;3B  
Ndmki 7A  
 
评价一下你浏览此帖子的感受

精彩

感动

搞笑

开心

愤怒

无聊

灌水
描述
快速回复

您目前还是游客,请 登录注册
温馨提示:欢迎交流讨论,请勿纯表情、纯引用!
认证码:
验证问题:
10+5=?,请输入中文答案:十五