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

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

级别: 终身会员
发帖
3743
铜板
8
人品值
493
贡献值
9
交易币
0
好评度
3746
信誉值
0
金币
0
所在楼道
代码审查是消灭Bug最重要的方法之一,这些审查在大多数时候都特别奏效。由于代码审查本身所针对的对象,就是俯瞰整个代码在测试过程中的问题和Bug。并且,代码审查对消除一些特别细节的错误大有裨益,尤其是那些能够容易在阅读代码的时候发现的错误,这些错误往往不容易通过机器上的测试识别出来。本文就常见的Java代码中容易出现的问题提出一些建设性建议,以便您在审查代码的过程中注意到这些常见的细节性错误。 398}a!XM  
`<y2l94tL  
k=w%oqpN  
  通常给别人的工作挑错要比找自己的错容易些。别样视角的存在也解释了为什么作者需要编辑,而运动员需要教练的原因。不仅不应当拒绝别人的批评,我们应该欢迎别人来发现并指出我们的编程工作中的不足之处,我们会受益匪浅的。 e][U ;  
M*8Ef^-U`t  
s+C&\$E  
(tx6U.Oy  
 正规的代码审查(code inspection)是提高代码质量的最强大的技术之一,代码审查?由同事们寻找代码中的错误?所发现的错误与在测试中所发现的错误不同,因此两者的关系是互补的,而非竞争的。 gAgzM?A1(  
8>6+]]O  
!4]w b!F  
V8`t7[r  
  如果审查者能够有意识地寻找特定的错误,而不是靠漫无目的的浏览代码来发现错误,那么代码审查的效果会事半功倍。在这篇文章中,我列出了11个Java编程中常见的错误。你可以把这些错误添加到你的代码审查的检查列表(checklist)中,这样在经过代码审查后,你可以确信你的代码中不再存在这类错误了。 )@\= pE.H  
C:l /%   
e6QUe.S  
. sgV  
  一、常见错误1# :多次拷贝字符串 -+#\WB{AI  
<8+.v6DCd  
C:0Ra^i ?L  
DE^{8YX,  
  测试所不能发现的一个错误是生成不可变(immutable)对象的多份拷贝。不可变对象是不可改变的,因此不需要拷贝它。最常用的不可变对象是String。 K.",=\53  
HPg@yx"U  
80&JEtRh  
%W+*)u72(  
  如果你必须改变一个String对象的内容,你应该使用StringBuffer。下面的代码会正常工作: !d&K,k  
;6U=fBp7<  
K82pWpR  
O9dIobu4  
String s = new String ("Text here"); kDS4 t?Ig  
 l}5@6;}  
Q(T)s  
pI,QkDJ0  
  但是,这段代码性能差,而且没有必要这么复杂。你还可以用以下的方式来重写上面的代码: r|{h7'  
4! XB?-.  
 #dtYa  
k/u6Cw0/  
String temp = "Text here"; o;D87E6Z  
String s = new String (temp); zVd2kuI&?  
U_wn/wcLS  
S}cpYjnH8  
jY(' ?3  
  但是这段代码包含额外的String,并非完全必要。更好的代码为: fJH09:@^%  
ltO:./6v  
YRfs8I^rg  
}'b 3'/MJ  
String s = "Text here"; _b&Mrd  
J;Xh{3[vO  
*[wy- fu  
S>/p6}3]  
  二、常见错误2#: 没有克隆(clone)返回的对象 M-e!F+d{od  
^}8(o  
.a8N 5{`  
J3Qv|w [3Y  
  封装(encapsulation)是面向对象编程的重要概念。不幸的是,Java为不小心打破封装提供了方便??Java允许返回私有数据的引用(reference)。下面的代码揭示了这一点: F@& R"-  
p&>*bF,  
\A6MVMF8  
q?nXhUD  
import java.awt.Dimension; o )G'._  
/***Example class.The x and y values should never*be negative.*/ kn^RS1m  
public class Example{ +%OINMo.A  
  private Dimension d = new Dimension (0, 0); O={4 >>F  
  public Example (){ } \3-XXq  
!\'7j-6  
  /*** Set height and width. Both height and width must be nonnegative * or an exception is thrown.*/ +?w 7Nm`  
  public synchronized void setValues (int height,int width) throws IllegalArgumentException{ GLp2 ?fon  
   if (height < 0 || width < 0) m$ )yd~  
    throw new IllegalArgumentException(); h q6B pE  
    d.height = height; jr|(K*;  
     d.width = width; r/$+'~apTk  
  } .0:BgM  
rjo/-910  
  public synchronized Dimension getValues(){ D^baXp8  
   // Ooops! Breaks encapsulation J}c57$Z  
   return d; wZJpSkcEx  
  } ug'I:#@2  
} XZEawJ0  
IEfzu L<v  
2?u>A3^R  
n (7m  
  Example类保证了它所存储的height和width值永远非负数,试图使用setValues()方法来设置负值会触发异常。不幸的是,由于getValues()返回d的引用,而不是d的拷贝,你可以编写如下的破坏性代码: gPSUxE `O.  
=Mzg={)v  
cv=nGFx6  
l"5$6h  
Example ex = new Example(); s:'M[xI  
Dimension d = ex.getValues(); ZR.1SA0x?O  
d.height = -5; [^EU'lewnW  
d.width = -10; w,bILv)  
/;-KWu+5=  
|NJe4lw+?  
L(\sO=t  
  现在,Example对象拥有负值了!如果getValues() 的调用者永远也不设置返回的Dimension对象的width 和height值,那么仅凭测试是不可能检测到这类的错误。 &tB|l_p_-p  
4EQ7OGU  
*Z>Yv37P  
 Zf68 EB  
  不幸的是,随着时间的推移,客户代码可能会改变返回的Dimension对象的值,这个时候,追寻错误的根源是件枯燥且费时的事情,尤其是在多线程环境中。 'b:e`2fl  
;2Db/"`t  
bW(+Aw=O  
P~Q5d&1SO  
  更好的方式是让getValues()返回拷贝: {hGr`Rh  
! E` Tt[  
vA2@Db}  
6F6[w?   
public synchronized Dimension getValues(){ 5cO}Jp%PA  
return new Dimension (d.x, d.y); @kvgq 0ab  
} $#2ik~]>  
.;yy= Rj  
QWH1xId  
O<Qa1Ow7f  
  现在,Example对象的内部状态就安全了。调用者可以根据需要改变它所得到的拷贝的状态,但是要修改Example对象的内部状态,必须通过setValues()才可以。 BF@(`D&>  
blNE$X+0|  
$e& ( ncM  
l>`N+ pZ$  
  三、常见错误3#:不必要的克隆 R $HI JM  
j/4N  
_IuEa\>  
},KY9w  
  我们现在知道了get方法应该返回内部数据对象的拷贝,而不是引用。但是,事情没有绝对: /e1m1B  
gP"p7\ (  
)X@Obg  
@'C f<wns  
/*** Example class.The value should never * be negative.*/ *vc=>AEc  
public class Example{ * t6 XU  
  private Integer i = new Integer (0); 8ar2N)59  
  public Example (){ } .F:qJ6E  
b#bdz1@s  
  /*** Set x. x must be nonnegative* or an exception will be thrown*/ iDt^4=`  
  public synchronized void setValues (int x) throws IllegalArgumentException{ vDZhoD=VR  
   if (x < 0) R$' 4 d  
    throw new IllegalArgumentException(); m^rgzx19?  
    i = new Integer (x); Y:[WwX|  
  } W7>4-gk  
sP$bp Z}  
  public synchronized Integer getValue(){ W.iL!x.B@  
   // We can’t clone Integers so we makea copy this way. R#i|n< x  
   return new Integer (i.intValue()); 0@d)DLM?  
  } xx0s`5  
} qg#TE-Y`  
lc>)7UF  
A`Q'I$fj  
Qmle0ae  
  这段代码是安全的,但是就象在错误1#那样,又作了多余的工作。Integer对象,就象String对象那样,一旦被创建就是不可变的。因此,返回内部Integer对象,而不是它的拷贝,也是安全的。 Uhfm@1 cz&  
'bGL@H  
i#$9>X  
-FytkM^]6  
  方法getValue()应该被写为: yn<H^c  
FL% GW:  
CnruaN@  
?jbE3fW  
public synchronized Integer getValue(){ *( YtO  
// ’i’ is immutable, so it is safe to return it instead of a copy. +N5#EpW  
return i; 2ME"=! &5  
} 0JQy-hpF  
:_JZn`Cab  
EbSH)aR  
}c1Vu  
  Java程序比C++程序包含更多的不可变对象。JDK 所提供的若干不可变类包括: nkTH#WTfR  
-NtT@ +AE  
*T"JO |  
c|3%0=,`  
  ?Boolean Mnyg:y*=  
   ?Byte T0s7aw[zm  
   ?Character %^[45e  
   ?Class S>O fUrt  
   ?Double 0Ge*\Q  
   ?Float 8*kZ.-T B  
   ?Integer Y,RED5]t  
   ?Long v39`ct=e  
   ?Short ?(Q" y\  
   ?String tt%Zwf  
   ?大部分的Exception的子类 r?Jxl<  
kCfSF%W&  
qH!}oPeU'  
A!Zjcp|  
 四、常见错误4# :自编代码来拷贝数组 V#[I/D  
UMwB.*  
"r @RDw   
r/1:!Vu(  
  Java允许你克隆数组,但是开发者通常会错误地编写如下的代码,问题在于如下的循环用三行做的事情,如果采用Object的clone方法用一行就可以完成: gS4zX>rqe  
A`<#}~A  
.o91^jt  
 hLFf  
public class Example{ GHj1G,L@\  
  private int[] copy; *@o@>  
  /*** Save a copy of ’data’. ’data’ cannot be null.*/ 7Ipt~K}  
  public void saveCopy (int[] data){ E*ybf'  
   copy = new int[data.length]; vpXC5|9U  
   for (int i = 0; i < copy.length; ++i) >JwdVy^  
    copy = data; r@FdxsCnGM  
  } 9 tkj:8_  
} K];nM}<  
rB;` &)-  
r|4jR6%<'m  
BM=`zGh"  
  这段代码是正确的,但却不必要地复杂。saveCopy()的一个更好的实现是: `?LQd2p  
ta"/R@ k*  
SY|r'8Z%Q  
qJ|ByZ.N+  
void saveCopy (int[] data){ [1B F8:  
  try{ J9S9r ir&  
   copy = (int[])data.clone(); D}'g4Ag  
  }catch (CloneNotSupportedException e){ mj5$ 2J  
   // Can’t get here. Ol H{!  
  } c+?L?s`"  
} },'hhj]O  
-/|O*oZ  
I7TdBe-  
2Fi>nJ  
  如果你经常克隆数组,编写如下的一个工具方法会是个好主意: 0/hX3h  
*I%r   
wGa0w*$  
^;+lsEW  
static int[] cloneArray (int[] data){ W7.O(s,32  
  try{ %HSS x+2oR  
   return(int[])data.clone(); uXq?Z@af|f  
  }catch(CloneNotSupportedException e){ LqIMU4Ex  
   // Can’t get here. d,[.=Jqv[  
  } b9ysxuUdS  
} zrk/}b0j  
RrKAgw  
3S1V^C-eBx  
>"?HbR9  
  这样的话,我们的saveCopy看起来就更简洁了: G~_5E]8  
2+" =i/8  
z"o;|T:  
u_.V]Rjc  
void saveCopy (int[] data){ L(TO5Y]  
  copy = cloneArray ( data); #b\&Md|;  
} >C"cv^%c  
2n+j.  
Gp9 >R~$  
1Uz'= a  
  五、常见错误5#:拷贝错误的数据 vM~/|)^0sW  
?2[=llS4  
^|xj.  
W093rNF~  
  有时候程序员知道必须返回一个拷贝,但是却不小心拷贝了错误的数据。由于仅仅做了部分的数据拷贝工作,下面的代码与程序员的意图有偏差: Na: M1Uhb  
j>gO]*BX~  
Y>t*L#i  
?"+' OOqik  
import java.awt.Dimension; /eDah3%d  
/*** Example class. The height and width values should never * be gJi11^PK  
negative. */ I@cKiB  
public class Example{ *{W5QEa  
  static final public int TOTAL_VALUES = 10; w3Aq[1U0  
  private Dimension[] d = new Dimension[TOTAL_VALUES]; ^T'+dGU`  
  public Example (){ } j^KM   
^i&Qr+v  
  /*** Set height and width. Both height and width must be nonnegative * or an exception will be thrown. */ )ZzwD]  
  public synchronized void setValues (int index, int height, int width) throws IllegalArgumentException{  Cih}  
   if (height < 0 || width < 0) vq$%Ug/B  
    throw new IllegalArgumentException(); \F,?ptu  
    if (d[index] == null) ;1S{xd*^N  
     d[index] = new Dimension(); ]w%7/N0R  
     d[index].height = height; c}Jy'F7&f  
     d[index].width = width; V)R-w`  
  } GK/a^[f+'l  
  public synchronized Dimension[] getValues() o]n5pZ\\W<  
   throws CloneNotSupportedException{ ,8o]XFOr  
    return (Dimension[])d.clone(); R8EDJ2u#  
  } gv `jeN  
} GEA@AD=^f  
%xxe U  
Bp^>R`,  
vtR<(tOu@  
  这儿的问题在于getValues()方法仅仅克隆了数组,而没有克隆数组中包含的Dimension对象,因此,虽然调用者无法改变内部的数组使其元素指向不同的Dimension对象,但是调用者却可以改变内部的数组元素(也就是Dimension对象)的内容。方法getValues()的更好版本为: vb: '%^v  
<| |Lj  
*[ Wh9 ,H  
r!Eo8C  
public synchronized Dimension[] getValues() throws CloneNotSupportedException{ ( NjX?^  
  Dimension[] copy = (Dimension[])d.clone(); {ZbeF#*"  
  for (int i = 0; i < copy.length; ++i){ ~FZLA}  
   // NOTE: Dimension isn’t cloneable. St|sUtj<r  
   if (d != null) [lS'GszA  
    copy = new Dimension (d.height, d.width); 5 W(iU  
  } -iBu:WyY$  
  return copy; mwbkXy;8  
}  .^@+$}   
WSDNTfpI  
_<;#=l  
wVE"nN#  
  在克隆原子类型数据的多维数组的时候,也会犯类似的错误。原子类型包括int,float等。简单的克隆int型的一维数组是正确的,如下所示: SZG8@ !_}7  
BOL_kp"   
3I:DL#f  
%Tsefs?_  
public void store (int[] data) throws CloneNotSupportedException{ FD|R4 V*3  
  this.data = (int[])data.clone(); GD[~4G  
  // OK n$`Nx\v  
} H=X>o.iVqi  
zF)_t S  
m>:%[vm  
ddnWr"_  
  拷贝int型的二维数组更复杂些。Java没有int型的二维数组,因此一个int型的二维数组实际上是一个这样的一维数组:它的类型为int[]。简单的克隆int[][]型的数组会犯与上面例子中getValues()方法第一版本同样的错误,因此应该避免这么做。下面的例子演示了在克隆int型二维数组时错误的和正确的做法: }C" #b\A2  
ct~lt'L\  
)yJeh  
J)(]cW.  
public void wrongStore (int[][] data) throws CloneNotSupportedException{ b${Kj3(  
  this.data = (int[][])data.clone(); // Not OK! 1}[\@n+b  
} H _3gVrP_  
public void rightStore (int[][] data){ Syp|s3u;  
  // OK! h^hEyrJw  
  this.data = (int[][])data.clone(); wk9tJ#}  
  for (int i = 0; i < data.length; ++i){ U45/%?kE)  
   if (data != null) 2d.I3z:[  
    this.data = (int[])data.clone(); 7 UQD02  
  } = 1}-]ctVn  
} 9%zR ? u  
DVTzN(gO*~  
4i~;Ql  
qh.c#t  
i;|% hDNWA  
   六、常见错误6#:检查new 操作的结果是否为null ACyQsmqm:  
r{%NMj  
iZSj T"l^  
2vWkAC;   
  Java编程新手有时候会检查new操作的结果是否为null。可能的检查代码为: ` |]6<<'iW  
2"__jp:(  
rEAPlO.Yp  
JH)&Ca>S  
Integer i = new Integer (400); r4D66tF  
if (i == null) _R5^4-Qe  
throw new NullPointerException(); ;F5B)&/B  
,\=u(Y\I[  
zLxWyPM0;  
? erDP8  
  检查当然没什么错误,但却不必要,if和throw这两行代码完全是浪费,他们的唯一功用是让整个程序更臃肿,运行更慢。 ^f`#8G7(  
]KuK\(\  
uxh4nyE  
ZgXh[UHQy  
  C/C++程序员在开始写java程序的时候常常会这么做,这是由于检查C中malloc()的返回结果是必要的,不这样做就可能产生错误。检查C++中new操作的结果可能是一个好的编程行为,这依赖于异常是否被使能(许多编译器允许异常被禁止,在这种情况下new操作失败就会返回null)。在java 中,new 操作不允许返回null,如果真的返回null,很可能是虚拟机崩溃了,这时候即便检查返回结果也无济于事。 'rR\H2b   
![vc/wuf  
 七、常见错误7#:用== 替代.equals F`u{'w:Hv  
wfZ 'T#1  
  在Java中,有两种方式检查两个数据是否相等:通过使用==操作符,或者使用所有对象都实现的.equals方法。原子类型(int, flosat, char 等)不是对象,因此他们只能使用==操作符,如下所示: zkd#vAY(A  
fi?[ e?|c@  
%pwm34  
MfL q h  
int x = 4; ^k)f oD  
int y = 5; kW,yZ.?f  
if (x == y) G*"N}M1)  
  System.out.println ("Hi"); Hb]7>[L  
// This ’if’ test won’t compile. kb%W3c9HO  
if (x.equals (y)) qp{NRNkQ  
  System.out.println ("Hi"); J,~)9Kh$  
n]o+KT\  
3Q;^X(Ml*  
Gg;#U`  
  对象更复杂些,==操作符检查两个引用是否指向同一个对象,而equals方法则实现更专门的相等性检查。 9J%>2AA  
S`6'~g  
n `n3[  
72{kig9c  
  更显得混乱的是由java.lang.Object 所提供的缺省的equals方法的实现使用==来简单的判断被比较的两个对象是否为同一个。 NK4ven7/  
`r]Cd {G  
/60[T@Mz  
;^*^ :L  
  许多类覆盖了缺省的equals方法以便更有用些,比如String类,它的equals方法检查两个String对象是否包含同样的字符串,而Integer的equals方法检查所包含的int值是否相等。 {:oZ&y)Ac  
-B4v1{An  
rmhCuY?f  
n!N;WL3k  
  大部分时候,在检查两个对象是否相等的时候你应该使用equals方法,而对于原子类型的数据,你用该使用==操作符。 +2?0]6EQ  
f0[xMn0Tu  
C>03P.s4c  
vj_oMmjKw  
  八、常见错误8#: 混淆原子操作和非原子操作 =~arj  
6F(yH4  
7"[lWC!As5  
m9q%l_  
  Java保证读和写32位数或者更小的值是原子操作,也就是说可以在一步完成,因而不可能被打断,因此这样的读和写不需要同步。以下的代码是线程安全(thread safe)的: vc!S{4bN  
Wh<lmC50(  
+(/Z=4;,[  
1a)_Lko  
public class Example{ 57j:Lw~   
  private int value; // More code here... O.4"h4{'  
  public void set (int x){ B{K'"uC  
   // NOTE: No synchronized keyword sEj:%`l|  
   this.value = x; lZua"Ju  
  } EjF}yuq[  
} &6:,2W&s  
YVaQ3o|!  
M\<!m^~  
UMX+h])#N  
  不过,这个保证仅限于读和写,下面的代码不是线程安全的: Mb9q<4  
P0Jd6"sS"  
9D| FqU |  
2$Ji4`p}S  
public void increment (){ Mu( Y6  
  // This is effectively two or three instructions: g SwG=e\  
  // 1) Read current setting of ’value’. QbNv+Eu5  
  // 2) Increment that setting. "TV.$s$.  
  // 3) Write the new setting back. C>u 3n^  
  ++this.value; >4VU  
} !'gz&3B~h  
"''<:K|  
=;a4 Dp  
V*m)h  
  在测试的时候,你可能不会捕获到这个错误。首先,测试与线程有关的错误是很难的,而且很耗时间。其次,在有些机器上,这些代码可能会被翻译成一条指令,因此工作正常,只有当在其它的虚拟机上测试的时候这个错误才可能显现。因此最好在开始的时候就正确地同步代码: XH2 SEeh  
#wd \&  
.;F+ QP0  
EGDE4n5>I  
public synchronized void increment (){ IL=v[)en4  
  ++this.value; =%u|8Ea*`  
} [!q&r(-K  
LAKZAi%O0  
iR_Syk`G*A  
'-S&i{H  
  九、常见错误9#:在catch 块中作清除工作 LWL>hd  
bc4x"]!  
__fR #D  
Y) h%<J  
  一段在catch块中作清除工作的代码如下所示: 8 (KfX%  
A{J1 n  
*0hiPj:  
)f!dG(\&#  
OutputStream os = null; '=~y'nPG7  
try{ p)^:~ ll  
  os = new OutputStream (); YJ^ lM\/<  
  // Do something with os here. 9HE(*S  
  os.close(); 4@V] zfu^Q  
}catch (Exception e){ 5>@uEebkv]  
  if (os != null) j'i42-Lt/p  
  os.close(); \l:g{GnoT  
} |Hm'.-   
?iLd5 Z  
,?`1ve_K<  
IeB6r+4|  
  尽管这段代码在几个方面都是有问题的,但是在测试中很容易漏掉这个错误。下面列出了这段代码所存在的三个问题: NslA/"*  
m3(T0.j0P  
-n *>zGc  
:]^P ^khK  
  1.语句os.close()在两处出现,多此一举,而且会带来维护方面的麻烦。 BXo|CITso  
uT/B}`md  
n2{{S(N  
"k<:a2R  
  2.上面的代码仅仅处理了Exception,而没有涉及到Error。但是当try块运行出现了Error,流也应该被关闭。 &a=e=nR5  
AMd)d^;  
xN m32~  
<s >/< kW:  
  3.close()可能会抛出异常。 D'`"_  
Z&w^9;30P  
^CPfo/!  
0Mu8ZVI{  
  上面代码的一个更优版本为: o$ce1LO?|N  
KF_Wu}q d  
^A[`NYK  
F!&pENQ  
OutputStream os = null; 2]3HX3  
try{ ~Ex.Yp8.  
  os = new OutputStream (); :dguQ|e  
  // Do something with os here. b!X"2'  
}finally{ EOX_[ek7  
  if (os != null) 0['"m^l0S  
   os.close(); @pG lWw9*  
} iEviH>b5  
zf,%BI[Hr  
&@HNz6KO  
7+a%ehwU  
  这个版本消除了上面所提到的两个问题:代码不再重复,Error也可以被正确处理了。但是没有好的方法来处理第三个问题,也许最好的方法是把close()语句单独放在一个try/catch块中。 [u9JL3  
~'2r&?=\  
bk wa{V  
.W :  
  十、常见错误10#: 增加不必要的catch 块 WYC1rfd=  
NVJ&C]H6  
Nr24[e G>d  
0R}Sw[M.  
  一些开发者听到try/catch块这个名字后,就会想当然的以为所有的try块必须要有与之匹配的catch块。 >_`D3@Rz  
[DxefYyI  
ZSRR lkU  
"P'&+dH8  
  C++程序员尤其是会这样想,因为在C++中不存在finally块的概念,而且try块存在的唯一理由只不过是为了与catch块相配对。 `8(h,aj;  
-#wVtXaSc  
}11`98>B6:  
i.t9jN  
  增加不必要的catch块的代码就象下面的样子,捕获到的异常又立即被抛出: 7y)=#ZG'R  
4`6c28K0?  
3_MS'&M  
V[Rrst0yo  
try{ +lW}ixt  
  // Nifty code here adI!W-/R:  
}catch(Exception e){ $% Ci8p  
  throw e; qo6LC>Qg  
}finally{ >&;>PZBPCO  
  // Cleanup code here l#b|@4:I  
} +`*qlP;  
7w Q+giu  
xegQRc  
fJWxJSdi  
  不必要的catch块被删除后,上面的代码就缩短为: "%E-X:Il#  
'B3Wza.  
3e%l8@R@  
`z}vONXpAX  
try{ Jq$_=X&  
  // Nifty code here c6,s+^^  
}finally{ +9zJlL^A%  
  // Cleanup code here VW9>xVd4  
} UZje>. ~?  
{}_Nep/;  
oWp}O?  
ZU|6jI}  
  常见错误11#;没有正确实现equals,hashCode,或者clone 等方法 dP$8JI{  
)'[x)q  
"{A*(.  
S jVsF1d_  
  方法equals,hashCode,和clone 由java.lang.Object提供的缺省实现是正确的。不幸地是,这些缺省实现在大部分时候毫无用处,因此许多类覆盖其中的若干个方法以提供更有用的功能。但是,问题又来了,当继承一个覆盖了若干个这些方法的父类的时候,子类通常也需要覆盖这些方法。在进行代码审查时,应该确保如果父类实现了equals,hashCode,或者clone等方法,那么子类也必须正确。正确的实现equals,hashCode,和clone需要一些技巧。 ~=:2~$gsn  
UzgA26;  
0]T.Lh$3  
mRhd/|g*  
  小结 fvcW'T}r  
{f+N]Oo*  
v2hZq-q  
s\'y-UITi1  
  我在代码审查的时候至少遇到过一次这些错误,我自己也犯过其中的几个错误。好消息是只要你知道你在找什么错误,那么代码审查就很容易管理,错误也很容易被发现和修改。即便你找不到时间来进行正规的代码审查,以自审的方式把这些错误从你的代码中根除会大大节省你的调试时间。花时间在代码审查上是值得的。 p)B33Z zC  
6a4'xq7  
 
评价一下你浏览此帖子的感受

精彩

感动

搞笑

开心

愤怒

无聊

灌水
描述
快速回复

您目前还是游客,请 登录注册
如果您在写长篇帖子又不马上发表,建议存为草稿
认证码:
验证问题:
10+5=?,请输入中文答案:十五