一般来说,最好能重用对象而不是每次需要的时候创建一个相同功能的新对象。
一些无意识创建不必要的对象的情况:
String实例
String s=new String("gyx");
上述语句每次被执行时都创建一个新的String实例,改进后为String s="gyx";
自动装箱机制
Long sum = 0L;
long start = System.currentTimeMillis();
for (int i = 0; i< Integer.MAX_VALUE; i++) {
sum += i;
}
long end = System.currentTimeMillis();
System.out.println(sum + " 用时: " + (end - start) + "ms");
上述把sum声明成Long,那么每次加i时都会把i自动装箱成Long。
方法内重复创建常量对象
-
对于同时提供了静态工厂方法和构造器的不可变类,通常使用静态工厂方法而不是构造器以避免创建不必要的对象。
-
除了重用不可变的对象之外,也可以重用那些已知不会被修改的可变对象。举例如下:
一个类中设计可变的Date对象,它们的值一旦被计算之后就不会再变化:下面这个方法用来检测一个人是否为一个生育高峰期出生的小孩。
public class Person {
private final Date brithDate;
public Person(Date brithDate) {
this.brithDate = brithDate;
}
public boolean isBabyBoomer() {
//Unnecessary allocation of expensive object
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 brithDate.compareTo(boomStart) >= 0 &&
brithDate.compareTo(boomEnd) < 0;
}
}
- isBabyBoomer方法每次被调用时,都会新建一个Calendar、一个TimeZone和两个Date实例,下面使用一个静态初始化器避免了这种情况:
public class Person {
private final Date brithDate;
private final static Date BOOM_START;
private final static Date BOOM_END;
public Person(Date brithDate) {
this.brithDate = brithDate;
}
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 brithDate.compareTo(BOOM_START) >= 0 &&
brithDate.compareTo(BOOM_END) < 0;
}
}