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

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

级别: 终身会员
发帖
3743
铜板
8
人品值
493
贡献值
9
交易币
0
好评度
3746
信誉值
0
金币
0
所在楼道
代码审查是消灭Bug最重要的方法之一,这些审查在大多数时候都特别奏效。由于代码审查本身所针对的对象,就是俯瞰整个代码在测试过程中的问题和Bug。并且,代码审查对消除一些特别细节的错误大有裨益,尤其是那些能够容易在阅读代码的时候发现的错误,这些错误往往不容易通过机器上的测试识别出来。本文就常见的Java代码中容易出现的问题提出一些建设性建议,以便您在审查代码的过程中注意到这些常见的细节性错误。 9^*RK6  
HI5NWdfRl  
uda++^y:  
  通常给别人的工作挑错要比找自己的错容易些。别样视角的存在也解释了为什么作者需要编辑,而运动员需要教练的原因。不仅不应当拒绝别人的批评,我们应该欢迎别人来发现并指出我们的编程工作中的不足之处,我们会受益匪浅的。 Cd'D ~'=  
_ZRmD\_t  
J^8j|%h%e  
#S7oW@  
 正规的代码审查(code inspection)是提高代码质量的最强大的技术之一,代码审查?由同事们寻找代码中的错误?所发现的错误与在测试中所发现的错误不同,因此两者的关系是互补的,而非竞争的。 >LPb>t5%p  
Fyvo;1a  
- (s0f  
*f+s  
  如果审查者能够有意识地寻找特定的错误,而不是靠漫无目的的浏览代码来发现错误,那么代码审查的效果会事半功倍。在这篇文章中,我列出了11个Java编程中常见的错误。你可以把这些错误添加到你的代码审查的检查列表(checklist)中,这样在经过代码审查后,你可以确信你的代码中不再存在这类错误了。 uEgR>X>  
o)I)I/v  
1Ek3^TOv7  
u7e$Mq  
  一、常见错误1# :多次拷贝字符串 VxY]0&sq  
3,p!Fun:r  
S9dx rm?  
rmg\Pa8W>  
  测试所不能发现的一个错误是生成不可变(immutable)对象的多份拷贝。不可变对象是不可改变的,因此不需要拷贝它。最常用的不可变对象是String。 ,i_+Z |Ls  
;f%@s1u  
=1[_#Moc6  
Zfs-M)  
  如果你必须改变一个String对象的内容,你应该使用StringBuffer。下面的代码会正常工作: GgxPpS<ne  
Z=% j|xE_  
~~yng-3)1  
uzp\V 39  
String s = new String ("Text here"); L@Rgiq|v-|  
+s#%\:Y M  
}+j B5z'w  
RLf-Rdx/  
  但是,这段代码性能差,而且没有必要这么复杂。你还可以用以下的方式来重写上面的代码: nWK8.&{.  
HxbzFu?h  
 %lj5Olj  
s_ZPo6p  
String temp = "Text here"; ~ZafTCa;  
String s = new String (temp); wH"9N+82M  
8L[+$g`  
yu_PZ"l  
E$%v);u  
  但是这段代码包含额外的String,并非完全必要。更好的代码为: CDJ@Tdp  
rl.K{Uad  
| V(sCF  
M8H hjoo  
String s = "Text here"; ]I*RuDv}  
]*NYuEgc  
i&DbZ=n2  
[Fh YQI  
  二、常见错误2#: 没有克隆(clone)返回的对象 ]v<8 l4p;  
#JYl%=#,  
@>2]zMFf  
:s_o'8z7L  
  封装(encapsulation)是面向对象编程的重要概念。不幸的是,Java为不小心打破封装提供了方便??Java允许返回私有数据的引用(reference)。下面的代码揭示了这一点: q%,86A>  
9swHa  
NFVu~t  
ltOS()[X  
import java.awt.Dimension; g:uVl;>  
/***Example class.The x and y values should never*be negative.*/ J *LPv9)  
public class Example{ L\mF[Kd#+T  
  private Dimension d = new Dimension (0, 0); ?EUg B\  
  public Example (){ } La6 9or   
<HnJD/g  
  /*** Set height and width. Both height and width must be nonnegative * or an exception is thrown.*/ O n0!>-b,  
  public synchronized void setValues (int height,int width) throws IllegalArgumentException{ }/J"/ T  
   if (height < 0 || width < 0) RrxbsG1HP  
    throw new IllegalArgumentException(); '$,yV f  
    d.height = height; fDYTupKXH  
     d.width = width; 7bYwh8  
  } gI7*zR4D  
{5VJprTbv  
  public synchronized Dimension getValues(){ {n>.Y -=  
   // Ooops! Breaks encapsulation W(s5mX,Kv  
   return d; 1*A^v  
  } @Yt394gA%\  
} I{w(`[Nxw*  
bR3Crz(9G  
i).Vu}W#S  
6!i`\>I]  
  Example类保证了它所存储的height和width值永远非负数,试图使用setValues()方法来设置负值会触发异常。不幸的是,由于getValues()返回d的引用,而不是d的拷贝,你可以编写如下的破坏性代码: #;99vwc  
gy?uk~p  
F7' MoH  
$j,$O>V  
Example ex = new Example(); f5//?ek  
Dimension d = ex.getValues(); '-myOM7  
d.height = -5; 6}Y==GP t  
d.width = -10; [!U%''  
H%vgPQ8  
6,4vs+(|\  
Lvt3S .l  
  现在,Example对象拥有负值了!如果getValues() 的调用者永远也不设置返回的Dimension对象的width 和height值,那么仅凭测试是不可能检测到这类的错误。 nHF66,7t  
,|O6<u9  
T}J)n5U}\  
0J?443A Y  
  不幸的是,随着时间的推移,客户代码可能会改变返回的Dimension对象的值,这个时候,追寻错误的根源是件枯燥且费时的事情,尤其是在多线程环境中。 @V>]95RX  
|./:A5_h  
:UT \L2 q=  
U _pPI$ =  
  更好的方式是让getValues()返回拷贝: OfrzmL<K  
v,opyTwG|  
P7>\j*U91{  
Tf=1p1!3  
public synchronized Dimension getValues(){ ku/vV+&O  
return new Dimension (d.x, d.y); mm_)=Ipj>  
} *_YH}U  
AxEdQRGk  
PP! /WX  
X\BFvSv8C  
  现在,Example对象的内部状态就安全了。调用者可以根据需要改变它所得到的拷贝的状态,但是要修改Example对象的内部状态,必须通过setValues()才可以。 N5W!(h)  
gb!0%*   
?6"U('y>n  
G5|nt#>  
  三、常见错误3#:不必要的克隆 v~x`a0  
F,as>X#  
cGs& Kn;h  
pzt<[;  
  我们现在知道了get方法应该返回内部数据对象的拷贝,而不是引用。但是,事情没有绝对: _x|R`1`  
>'#vC]@  
E<D^j^T  
N[-$*F,:_  
/*** Example class.The value should never * be negative.*/ pFs/ipZX^*  
public class Example{ 3w>1R>7  
  private Integer i = new Integer (0); C/ VHzV%q  
  public Example (){ } gcI<bY  
{oAD;m`  
  /*** Set x. x must be nonnegative* or an exception will be thrown*/ N G vb]  
  public synchronized void setValues (int x) throws IllegalArgumentException{ 3rMi:*?  
   if (x < 0) 7[ n |3  
    throw new IllegalArgumentException(); g?iZ RM  
    i = new Integer (x); Gv]94$'J9  
  } <k3KCt  
vH}VieU  
  public synchronized Integer getValue(){ ;TC]<N.YJT  
   // We can’t clone Integers so we makea copy this way. [ Y{  
   return new Integer (i.intValue()); SnX)&>B  
  } P_H2[d&/>D  
} ltrti.&  
w_"-rGV  
uzb|yV'B  
Ve9) ?=!  
  这段代码是安全的,但是就象在错误1#那样,又作了多余的工作。Integer对象,就象String对象那样,一旦被创建就是不可变的。因此,返回内部Integer对象,而不是它的拷贝,也是安全的。 %<8?$-[  
mYfHBW:  
OW6dK #CFt  
b7+(g [O  
  方法getValue()应该被写为: S.>fB7'(?=  
uMm`j?Y23q  
(I6Q"&h]  
NZG ^B/  
public synchronized Integer getValue(){ |F\fdB}?S:  
// ’i’ is immutable, so it is safe to return it instead of a copy. U:@tdH+A7  
return i; N2|NYDQs  
} yXIJeo"  
j"Ew)6j  
FQ 0 ;%Z  
K[?@nl?,z  
  Java程序比C++程序包含更多的不可变对象。JDK 所提供的若干不可变类包括: Wc m'E3c,  
}!r pH{y  
 `wIWK7i  
C2b<is=H:  
  ?Boolean a".iVf6y  
   ?Byte zRgGSxn  
   ?Character ZmkH55Cn  
   ?Class :_F 8O  
   ?Double t@ri`?0w  
   ?Float F_ -Xx"  
   ?Integer 1Ke9H!_P  
   ?Long dEI!r1~n  
   ?Short [_ uT+q3  
   ?String yK"HHdYTV  
   ?大部分的Exception的子类 "9X!Ewm"P  
vqVwo\oEdU  
RH7!3ye  
zFDtC-GF  
 四、常见错误4# :自编代码来拷贝数组 RZVZ#q(DU  
n'j}u  
:)4c_51 `  
tCRsaDK>  
  Java允许你克隆数组,但是开发者通常会错误地编写如下的代码,问题在于如下的循环用三行做的事情,如果采用Object的clone方法用一行就可以完成: A"qDc  
Z<=L  
"E4CQL'U  
T#:b  
public class Example{ F.@|-wq&  
  private int[] copy; ;gAL_/_  
  /*** Save a copy of ’data’. ’data’ cannot be null.*/ B7Zi|-F  
  public void saveCopy (int[] data){ +~:OUR*>  
   copy = new int[data.length]; CRiqY_gBf  
   for (int i = 0; i < copy.length; ++i) 2dB]Lw@s  
    copy = data; K:VZ#U(_  
  } B>S>t5$  
} zmu+un"\j  
u|\?6fz  
\J#&]o)Y  
 JJs*2y  
  这段代码是正确的,但却不必要地复杂。saveCopy()的一个更好的实现是: uvR l`"Y  
*c%{b3T_  
>[nR$8_J-l  
cdGBo4  
void saveCopy (int[] data){  V_e  
  try{ RU/SJ1wM"  
   copy = (int[])data.clone(); I\M }Dxpp  
  }catch (CloneNotSupportedException e){ ]Nssn\X7  
   // Can’t get here. ; bHS^  
  } ,h/l-#KS  
} f)Y~F/[$P  
:AQ9-&i/a-  
0`v-pL0|  
#Jp|Cb<qx  
  如果你经常克隆数组,编写如下的一个工具方法会是个好主意: n{{"+;oR  
r XBC M  
JrX. f  
ZzQLbCV  
static int[] cloneArray (int[] data){ ZCBF&.!  
  try{ !&.-{ _$  
   return(int[])data.clone(); i6P$>8jBQ-  
  }catch(CloneNotSupportedException e){ e^x%d[sU  
   // Can’t get here. '.gi@Sr5  
  } pp{p4Z   
} DvLwX1(l  
+7AH|v8  
CY*GCkH  
<$Sl%DoS  
  这样的话,我们的saveCopy看起来就更简洁了: O.\\)8xA  
4#:Eq=(W  
Jk7 Am-.0  
MZWv#;.]  
void saveCopy (int[] data){ <c<!|<x  
  copy = cloneArray ( data); fz8 41 <Y  
} B~@Gfb>`'  
.A_R6~::  
@SaxM4  
;n|%W,b-  
  五、常见错误5#:拷贝错误的数据 oIj -Y`92!  
=&Tuh}  
"(dI/}  
8GjETq%}  
  有时候程序员知道必须返回一个拷贝,但是却不小心拷贝了错误的数据。由于仅仅做了部分的数据拷贝工作,下面的代码与程序员的意图有偏差: u]`0QxvZ  
%.'oY%  
`ueOb  
je3Qq1  
import java.awt.Dimension; ;R<V-gab  
/*** Example class. The height and width values should never * be ,!PV0(F(  
negative. */ B&1E&Cv_8  
public class Example{ f#7=N{wm  
  static final public int TOTAL_VALUES = 10; S,avvY.U\  
  private Dimension[] d = new Dimension[TOTAL_VALUES]; 8TU(5:xJo  
  public Example (){ } L8Z@Dk7Y  
p-w:l*-`  
  /*** Set height and width. Both height and width must be nonnegative * or an exception will be thrown. */ yOAC<<Tzus  
  public synchronized void setValues (int index, int height, int width) throws IllegalArgumentException{ o Q*LP{M  
   if (height < 0 || width < 0) tGbx/$Y   
    throw new IllegalArgumentException(); voTP,R[}85  
    if (d[index] == null) V eY&pPQ  
     d[index] = new Dimension(); !"-.D4*r  
     d[index].height = height; 5j0 Ib>\  
     d[index].width = width; Fq o h!F  
  } }s6Veosl  
  public synchronized Dimension[] getValues() |YV> #l  
   throws CloneNotSupportedException{ OQKc_z'"  
    return (Dimension[])d.clone(); ,q7FK z{  
  } >p;&AaXkoG  
} ;KEie@Ry  
F?cwIE\J  
=*zde0T?l  
Rh$+9w  
  这儿的问题在于getValues()方法仅仅克隆了数组,而没有克隆数组中包含的Dimension对象,因此,虽然调用者无法改变内部的数组使其元素指向不同的Dimension对象,但是调用者却可以改变内部的数组元素(也就是Dimension对象)的内容。方法getValues()的更好版本为: y7rT[f/J  
wf\7sz  
p&)d]oV>  
kd]CV7(7  
public synchronized Dimension[] getValues() throws CloneNotSupportedException{ iC>%P&|-)|  
  Dimension[] copy = (Dimension[])d.clone(); .=d40m  
  for (int i = 0; i < copy.length; ++i){ PyK!Cyq  
   // NOTE: Dimension isn’t cloneable. \IudS{ .?;  
   if (d != null) BpX`49  
    copy = new Dimension (d.height, d.width); fBz|-I:k +  
  } $ e,r>tgD  
  return copy; j+q)  
} B%k C>J  
` vFDO$K  
02NVdpo[wU  
4sBvW  
  在克隆原子类型数据的多维数组的时候,也会犯类似的错误。原子类型包括int,float等。简单的克隆int型的一维数组是正确的,如下所示: guf*>qNr  
)^"V}z t  
Dfc% jWbA  
2+C:Em0yI  
public void store (int[] data) throws CloneNotSupportedException{ gX(Xj@=(&  
  this.data = (int[])data.clone(); 0M&~;`W}  
  // OK  '.>y'=  
} gN7 3)uJ0  
)54a' Hp  
kUT^o  
X=lsuKREZ  
  拷贝int型的二维数组更复杂些。Java没有int型的二维数组,因此一个int型的二维数组实际上是一个这样的一维数组:它的类型为int[]。简单的克隆int[][]型的数组会犯与上面例子中getValues()方法第一版本同样的错误,因此应该避免这么做。下面的例子演示了在克隆int型二维数组时错误的和正确的做法: i3d 2+N`  
~ F-lO1  
#`K{vj  
ue@W@pj  
public void wrongStore (int[][] data) throws CloneNotSupportedException{ jt9- v-  
  this.data = (int[][])data.clone(); // Not OK! U}k@%m,  
} oR,zr  
public void rightStore (int[][] data){ _iEnS4$A8  
  // OK! ;volBfv  
  this.data = (int[][])data.clone(); & BY\h:  
  for (int i = 0; i < data.length; ++i){ %4V$')rek  
   if (data != null) kt\,$.v8  
    this.data = (int[])data.clone(); EA9.?F  
  } Oo FMOlb.Z  
} T}29(xz-(h  
XZ3fWcw[  
6%:~.ZfN  
'Nuy/\[{\  
P{:Zxli0  
   六、常见错误6#:检查new 操作的结果是否为null 2mMi=pv9  
:PY6J}:&#  
1CSGG'J]E  
[u^ fy<jdp  
  Java编程新手有时候会检查new操作的结果是否为null。可能的检查代码为: {.[EXMX  
G -K{  
^;9l3P{  
ur=:Ha  
Integer i = new Integer (400); mW+5I-~  
if (i == null) XzqB=iX  
throw new NullPointerException(); YktZXc?iI<  
x>tm[k  
bmi",UZ:F  
yHlQKI  
  检查当然没什么错误,但却不必要,if和throw这两行代码完全是浪费,他们的唯一功用是让整个程序更臃肿,运行更慢。 11Qi _T\  
F?6Q(mRl  
(NDC9Lls  
J4U_utp  
  C/C++程序员在开始写java程序的时候常常会这么做,这是由于检查C中malloc()的返回结果是必要的,不这样做就可能产生错误。检查C++中new操作的结果可能是一个好的编程行为,这依赖于异常是否被使能(许多编译器允许异常被禁止,在这种情况下new操作失败就会返回null)。在java 中,new 操作不允许返回null,如果真的返回null,很可能是虚拟机崩溃了,这时候即便检查返回结果也无济于事。 G51-CLM,  
7/k7V)  
 七、常见错误7#:用== 替代.equals /"m#mh L  
?z6K/'?  
  在Java中,有两种方式检查两个数据是否相等:通过使用==操作符,或者使用所有对象都实现的.equals方法。原子类型(int, flosat, char 等)不是对象,因此他们只能使用==操作符,如下所示: B!8X?8D  
8faT@J'e;  
$ <C",&  
UL#:!J/34  
int x = 4; 2Oyw#1tdn  
int y = 5; ["Tro;K#  
if (x == y) #CAZ}];Qx  
  System.out.println ("Hi"); 55UPd#E'  
// This ’if’ test won’t compile. K :+q9;g  
if (x.equals (y)) Bt5 P][<  
  System.out.println ("Hi"); WPlf8* -fQ  
/vi Ic %=  
~Cw7.NA{3  
Kng=v~)N'  
  对象更复杂些,==操作符检查两个引用是否指向同一个对象,而equals方法则实现更专门的相等性检查。 o"z;k3(i$7  
}Qe(6'l_  
A:2CP&*  
XqhrQU|wM  
  更显得混乱的是由java.lang.Object 所提供的缺省的equals方法的实现使用==来简单的判断被比较的两个对象是否为同一个。 P>)J:.tr0  
r!eW]M  
8t, &dq  
RW1+y/#%P  
  许多类覆盖了缺省的equals方法以便更有用些,比如String类,它的equals方法检查两个String对象是否包含同样的字符串,而Integer的equals方法检查所包含的int值是否相等。 v6Y[_1  
2G!z/OAj  
{ D|ST2:E  
X&5N 89  
  大部分时候,在检查两个对象是否相等的时候你应该使用equals方法,而对于原子类型的数据,你用该使用==操作符。 Q=vo5)t   
br 3-.g  
ycki0&n3  
,`!lZ| U  
  八、常见错误8#: 混淆原子操作和非原子操作 02tN=}Cj)  
-aE,KQ  
F9r/ M"5  
F$|:'#KN  
  Java保证读和写32位数或者更小的值是原子操作,也就是说可以在一步完成,因而不可能被打断,因此这样的读和写不需要同步。以下的代码是线程安全(thread safe)的: ;mz#$"(  
F2_'U' a  
<exyd6iI  
>SziRm>Y7  
public class Example{ y;4g>ma0  
  private int value; // More code here... 3 Fy C D4#  
  public void set (int x){ c&wiTvRV  
   // NOTE: No synchronized keyword w9RBT(u  
   this.value = x; #.fJ M:"tG  
  } Y8IC4:EO  
} D)l\zs%ie  
vlZmmQeJm  
[q_62[-X  
p1i}fGS  
  不过,这个保证仅限于读和写,下面的代码不是线程安全的:  cC|  
V*(x@pF  
ahCwA}  
%21|-B  
public void increment (){ Lc[TIX  
  // This is effectively two or three instructions: 02%~HBS  
  // 1) Read current setting of ’value’.  iycceZ  
  // 2) Increment that setting. OT=1doDp  
  // 3) Write the new setting back. ?MmQ'1N  
  ++this.value; )p>p3b g  
} q@XJ,e1A  
w'$>E4\   
+ug/%Iay{k  
~&F|g2:  
  在测试的时候,你可能不会捕获到这个错误。首先,测试与线程有关的错误是很难的,而且很耗时间。其次,在有些机器上,这些代码可能会被翻译成一条指令,因此工作正常,只有当在其它的虚拟机上测试的时候这个错误才可能显现。因此最好在开始的时候就正确地同步代码: _y>drvg  
$FX$nY  
gGBRfq>  
~UQ<8`@a  
public synchronized void increment (){ 5!$sQ@#}D  
  ++this.value; +opym!\  
} @|">j#0  
KSEKoHJo  
}U5$~, *p  
be]/ROP>H  
  九、常见错误9#:在catch 块中作清除工作 3&{6+A  
'W54 T  
F`(;@LO  
"cly99t  
  一段在catch块中作清除工作的代码如下所示: {%^4%Eco  
!;[cJbqnh  
|JWYsqJ0U  
n c~JAT# '  
OutputStream os = null; Oj_F1. r  
try{ DrAIQ7Jd  
  os = new OutputStream (); aj .7t =^  
  // Do something with os here. -a~n_Z>_  
  os.close(); ,D(Bg9C  
}catch (Exception e){ ePv`R'#  
  if (os != null) 9kqR-T|Q  
  os.close(); fZsw+PSy  
} vSoG] :1  
P ecZuv  
UGgo;e  
|]?W`KN0  
  尽管这段代码在几个方面都是有问题的,但是在测试中很容易漏掉这个错误。下面列出了这段代码所存在的三个问题: 8f)pf$v`   
fi~@J`  
+_S0  
c~OPH 0,  
  1.语句os.close()在两处出现,多此一举,而且会带来维护方面的麻烦。 /kRCCs8t}  
52Dgul  
5A|d hw   
#Hu# #x|  
  2.上面的代码仅仅处理了Exception,而没有涉及到Error。但是当try块运行出现了Error,流也应该被关闭。 0YfmAF$/B  
kX}sDvP3  
*mWl=J;u  
gN[t  
  3.close()可能会抛出异常。 J]S30&?  
S*J\YcqSC  
S>*i\OnI'  
t>*(v#WeZ  
  上面代码的一个更优版本为: 3W#E$^G_v  
!^0vi3I  
nec}grA  
Z0y~%[1X  
OutputStream os = null; g=qaq  
try{ 3b_/QT5!  
  os = new OutputStream (); 0CXXCa7!  
  // Do something with os here. `r3 klL,W'  
}finally{ bXXX-Xc  
  if (os != null)  QV\a f  
   os.close(); 6o9&FU  
} R;A8y  
\{~CO{II  
dvZlkMm   
k2,`W2] ^E  
  这个版本消除了上面所提到的两个问题:代码不再重复,Error也可以被正确处理了。但是没有好的方法来处理第三个问题,也许最好的方法是把close()语句单独放在一个try/catch块中。 eK*oV}U-k  
0_+ & [g}  
}-XZ1qr  
cwtlOg  
  十、常见错误10#: 增加不必要的catch 块 ~[og\QZX  
Vmh$c*TE  
vRf$#fBEQ  
7w8UnPuM  
  一些开发者听到try/catch块这个名字后,就会想当然的以为所有的try块必须要有与之匹配的catch块。 uW#s;1H.)  
s] ;P<  
D2gyn-]\  
um_J%v6ER  
  C++程序员尤其是会这样想,因为在C++中不存在finally块的概念,而且try块存在的唯一理由只不过是为了与catch块相配对。 " Qyi/r41  
*f>\X[wN  
Jq?zr]"A  
W$;qhB  
  增加不必要的catch块的代码就象下面的样子,捕获到的异常又立即被抛出: ,2 W=/,5A  
<&#]|HGc  
_j0xL{&&  
rbIYLVA+V  
try{ afD {w*[8  
  // Nifty code here p>3QW3<  
}catch(Exception e){ ?K2}<H-  
  throw e; cTRtMk%^  
}finally{ QUvSeNSp  
  // Cleanup code here %N(>B_t\  
} #9.%>1{6Y  
HJym|G>%?  
BtKor6ba  
Hy,""Py  
  不必要的catch块被删除后,上面的代码就缩短为: h7TkMt[l  
+Ig%h[1a  
ZUS5z+o  
Fo;:GX,b  
try{ ,RY;dX-#  
  // Nifty code here c|aX4=Z  
}finally{ W(4$.uZ)  
  // Cleanup code here Zby3.=.e  
} CQa8I2VF (  
cjO %X  
x{K"z4xbI  
dtfOFag4_  
  常见错误11#;没有正确实现equals,hashCode,或者clone 等方法 IO=$+c  
H<<t^,E^.t  
mT UoFXX[  
&=n/h5e0t&  
  方法equals,hashCode,和clone 由java.lang.Object提供的缺省实现是正确的。不幸地是,这些缺省实现在大部分时候毫无用处,因此许多类覆盖其中的若干个方法以提供更有用的功能。但是,问题又来了,当继承一个覆盖了若干个这些方法的父类的时候,子类通常也需要覆盖这些方法。在进行代码审查时,应该确保如果父类实现了equals,hashCode,或者clone等方法,那么子类也必须正确。正确的实现equals,hashCode,和clone需要一些技巧。 %xQ'i4`  
2e-bt@0t  
<%m1+%mA.  
-KfK~P3PF  
  小结 4e AMb  
>b=."i  
ONDO xXs  
UpE +WzY  
  我在代码审查的时候至少遇到过一次这些错误,我自己也犯过其中的几个错误。好消息是只要你知道你在找什么错误,那么代码审查就很容易管理,错误也很容易被发现和修改。即便你找不到时间来进行正规的代码审查,以自审的方式把这些错误从你的代码中根除会大大节省你的调试时间。花时间在代码审查上是值得的。 PUZcb+%]h  
.oT'(6#  
 
评价一下你浏览此帖子的感受

精彩

感动

搞笑

开心

愤怒

无聊

灌水
描述
快速回复

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