Dreamhead把他用大把银子买来的“Effective Java”借给我阅读,我真是很感动亚,我只能用行动来感谢Dreamhead了。^_^
24、需要时使用保护性拷贝
在学习这个item之前我们看看下面这段“危险的”代码(改编自书中例子):
Considering the following code:
//Period.java
import java.util.Date;
public final class Period {
private final Date start;
private final Date end;
public Period(Date start , Date end){
//do some checking , end should be later than start
this.start = start;
this.end = end;
}
public Date start(){
return this.start;
}
public Date end(){
return this.end;
}
//remainder omitted…
public static void main(String[] args) {
Date start = new Date(2004 , 11 , 28);
Date end = new Date(2004 , 11 , 30);
Period p = new Period(start , end);
System.out.println("Normal period:");
System.out.println(p.start());
System.out.println(p.end());
//danger code part1
System.out.println("Unnormal period part1:");
end.setMonth(4);
System.out.println(p.start());
System.out.println(p.end());
//danger code part2
System.out.println("Unnormal period part2:");
p.end().setMonth(5);
System.out.println(p.start());
System.out.println(p.end());
}
}
//output:
Normal period:
Wed Dec 28 00:00:00 CST 3904
Fri Dec 30 00:00:00 CST 3904
Unnormal period part1:
Wed Dec 28 00:00:00 CST 3904
Mon May 30 00:00:00 CST 3904
Unnormal period part2:
Wed Dec 28 00:00:00 CST 3904
Thu Jun 30 00:00:00 CST 3904
Period类声称提供一段不可改变的时间段,不过从上面的输出结果来看我们可以轻易的修改这个时间段。为了使上面的Period类真正成为非可变类,我们需要进行“defensive copy”。
Danger code part1提示我们“对构造函数的每个可变参数进行保护性拷贝”;
Danger code part2提示我们“对类中public方法返回的可变内部域进行保护性拷贝”;
修改如下:
public Period(Date start , Date end){
//do some checking , end should be later than start
this.start =new Date( start.getTime());
this.end = new Date(end.getTime());
}
public Date start(){
return (Date)this.start.clone();
}
public Date end(){
return (Date)this.end.clone();
}
修改后再运行,结果如下:
Output:
Wed Dec 28 00:00:00 CST 3904
Fri Dec 30 00:00:00 CST 3904
Unnormal period part1:
Wed Dec 28 00:00:00 CST 3904
Fri Dec 30 00:00:00 CST 3904
Unnormal period part2:
Wed Dec 28 00:00:00 CST 3904
Fri Dec 30 00:00:00 CST 3904
切记:在把类的内部组件的引用返回给客户之前,你应该返回一个保护性拷贝,如内部数组。
还有一点就是如果有可能你尽量使用immutable object作为你的内部组件,这样你就可以不必关心“defensive copy”的问题了。
25、小心设计方法的原型
在这个item中记住几个原则即可(这几个原则对我这个菜鸟来说用处还是蛮大的):
- 小心选择方法名字(可参考那本java developers almanac的索引来看看java中方法都是如何命名的)
- 对于参数类型,优先选用接口而不是类
- 小心使用function object , 原因不是主流的代码风格,难于理解。
34、通过接口引用对象
Item25中提到了“对于参数类型,优先选用接口”,更一般的讲我们应该用接口来引用对象。
书中的例子:
List subscribers = new Vector(); //good – use interface as type!
Vector subscribers = new Vector(); //bad-use class as type!
如果没有合适的接口,使用类来引用对象也是完全适合的。
其中有一种情况是当我们没有合适的接口时,我们尽量使用基类(往往是abstract class)来引用实现类(子类)的对象。
评论