Java中Sting很特别,有如下两种初始化方式:
(1).String s1 = “This isstring1”;
(2).String s2 = new String(“Thisis string2”);
第一种字符串初始化方式,当有多于一个字符串的内容相同情况,字符串内容会放在字符串缓冲池中,即字符串内容在内存中只有一份。
第二种字符串初始化方式,不论有没有字符串值相同,每次都会在内存堆中存储字符串的值。
如果一个方法中字符串的值都相同,调用100万次情况下第一种字符串初始化方式的内存占用率很低,性能非常高,而第二种方式的字符串初始化则会占用大量的内存.
看下面一个例子,直观感受创建不必要对象的性能危害:
public class Person{
private Date birthDate;
//判断是否是婴儿潮出生的人
public boolean isBabyBoomer(){
Calendar cal = Calendar.getInstance(TimeZone.getTimeZone(“GMT”));
//婴儿潮开始时间
cal.set(1946, Calendar.JANUARY, 1, 0, 0, 0);
Date boomStart = cal.getTime();
//婴儿潮结束时间
cal.set(1965, Calendar.JANUARY, 1, 0, 0, 0);
Date boomEnd = cal.getTime();
return birthDate.compareTo(boomStart) >= 0 && birthDate.compareTo(boomEnd) < 0;
}
}
每次调用isBabyBoomer()方式时,都需要创建Calendar,TimeZone,boomStart和boomEnd四个对象,测试调用该方法1000万次,大约耗时32秒。
Calendar,TimeZone,boomStart和boomEnd四个对象是所有调用者共用的对象,只需创建一份即可,改进之后代码如下:public class Person{
private Date birthDate;
private static final Date BOOM_START;
private static final Date BOOM_END;
static{
Calendar cal = Calendar.getInstance(TimeZone.getTimeZone(“GMT”));
//婴儿潮开始时间
cal.set(1946, Calendar.JANUARY, 1, 0, 0, 0);
BOOM_START = cal.getTime();
//婴儿潮结束时间
cal.set(1965, Calendar.JANUARY, 1, 0, 0, 0);
BOOM_END = cal.getTime();
}
//判断是否是婴儿潮出生的人
public boolean isBabyBoomer(){
return birthDate.compareTo(BOOM_START) >= 0 && birthDate.compareTo(BOOM_END) < 0;
}
}
经测试,调用1000万次,耗时大约130毫秒,性能提高250倍。
再看第二个例子,计算所有的int之和,代码如下:
public CalculateInt{
public static void main(String[] args){
Long sum = 0L;
for(long i = 0; i < Integer.MAX_VALUE; i++){
sum += i;
}
System.out.println(sum);
}
}
由于sum是Long类型,而不是原始类型的long,因此总共进行了2的31次方不必要的类型自动封装(将原始类型的long封装为Long),总共耗时大约43秒。
将sum声明的由Long类型改为原始类型的long,由于省去了2的31次方不必要的类型自动封装,程序大概耗时6秒,性能提高7倍。