看Effective java 略有所感,首先不要错觉的根据标题认为“创建对象的代价非常昂贵,我们应该要尽可能的避免创建对象”。相反,由于小对象的构造器只能做很少量的显示工作,所以,小对象的创建和回收动作是非常廉价的,特别是在现代的JVM实现上更是如此。通过创建附加小对象,提升程序的清晰性,简洁性和功能性,这通常是件好事情。
那接下来我们看看一个如何避免创建不必要的对象的代码,首先看看一个可能工作中经常会写的代码:这个类建立一个模型,其中有一个人,并有一个isBabyBoomer方法,用来检验这个人是否为一个“baby boomer”生育高峰期出生的小孩,换句话说,就是检验这个人是否出生于1946年至1964年期间
public class Person {
private final Date birthDate;
public Person(Date birthDate){
this.birthDate = birthDate;
}
public boolean isbabyBoomer(){
Calendar gmtCal = Calendar.getInstance(TimeZone.getTimeZone("GMT"));
gmtCal.set(1946, Calendar.JANUARY,1,0,0,0);
Date boomStart = gmtCal.getTime();
gmtCal.set(1965, Calendar.JANUARY,1,0,0,0);
Date boomEnd = gmtCal.getTime();
return birthDate.compareTo(boomStart)>=0 &&
birthDate.compareTo(boomEnd)<0;
}
}
很显然,上面的代码是一个反面的例子,千万不要这样做,应为isBabyBoomer每次被调用的时候,都会新建一个Calendar,一个TimeZone和两个Date实例,这个是不必要的,下面将看到一个用静态的初始化器(initializer)避免这种效率低下的情况
public class Person{
private final Date birthDate;
private static final Date BOOM_START;
private static final Date BOOM_END;
public Person(Date birthDate){
this.birthDate = birthDate;
}
static{
Calendar gmtCal = Calendar.getInstance(TimeZone.getTimeZone("GMT"));
gmtCal.set(1946, Calendar.JANUARY,1,0,0,0);
BOOM_START= gmtCal.getTime();
gmtCal.set(1965, Calendar.JANUARY,1,0,0,0);
BOOM_END = gmtCal.getTime();
}
public boolean isbabyBoomer(){
return birthDate.compareTo(BOOM_START)>=0 &&
birthDate.compareTo(BOOM_END)<0;
}
}
改进后的Person类只在初始化的时候创建Calendar,TimeZone和Date实例一次,而不是每次调用isBabyBoomer的时候都创建这些实例。如果isBabyBoomer方法被频繁调用,这种方法将会显著的提高性能。
节选自Effecttive java