在java中,使用最频繁、同时也是滥用最多的一个类或许就是java.lang.String,它也是导致代码性能低下最主要的原因之一。请考虑下面这个例子:
String s1 = "Testing String";
String s2 = "Concatenation Performance";
String s3 = s1 " " s2;
StringBuffer s = new StringBuffer();
s.append("Testing String");
s.append(" ");
s.append("Concatenation Performance");
String s3 = s.toString();
第二个代码片段用到了StringBuffer类(编译器在第一个片段中也将使用StringBuffer类),我们来分析一下StringBuffer类的默认构造函数,下面是它的代码:
public StringBuffer() { this(16); }
默认构造函数预设了16个字符的缓存容量。现在我们再来看看StringBuffer类的append()方法:public synchronized StringBuffer append(String str) {
if (str == null) {
str = String.valueOf(str);
}
int len = str.length();
int newcount = count len;
if (newcount >value.length) expandCapacity(newcount);
str.getChars(0, len, value, count);
count = newcount; return this;
}
在第二个代码片段中(以及在第一个代码片段的编译结果中),由于字符串追加操作的最后结果是“Testing String Concatenation Performance”,它有40个字符,StringBuffer的存储能力必须扩展两次,从而导致了两次代价昂贵的复制操作。因此,我们至少有一点可以做得比编译器更好,这就是分配一个初始存储容量大于或者等于40个字符的StringBuffer,如下所示:
StringBuffer s = new StringBuffer(45);
s.append("Testing String");
s.append(" ");
s.append("Concatenation Performance");
String s3 = s.toString();
String s = "";
int sum = 0;
for(int I=1; I<10; I ) {
sum = I;
s = s " " I ;
}
s = s "=" sum;
StringBuffer sb = new StringBuffer();
int sum = 0;
for(int I=1;
I<10; I ){
sum = I;
sb.append(I).append(" ");
}
String s = sb.append("=").append(sum).toString();
我们再来看看另外一个常用的Java类??java.util.Vector。简单地说,一个Vector就是一个java.lang.Object实例的数组。Vector与数组相似,它的元素可以通过整数形式的索引访问。但是,Vector类型的对象在创建之后,对象的大小能够根据元素的增加或者删除而扩展、缩小。请考虑下面这个向Vector加入元素的例子:
Object obj = new Object();
Vector v = new Vector(100000);
for(int I=0;
I<100000; I ) { v.add(0,obj); }
Object obj = new Object();
Vector v = new Vector(100000);
for(int I=0; I<100000; I ) { v.add(obj); }
假设要从前面的Vector删除所有元素,我们可以使用这种代码:
for(int I=0; I<100000; I )
{
v.remove(0);
}
for(int I=0; I<100000; I )
{
v.remove(v.size()-1);
}
v.removeAllElements();
假设Vector类型的对象v包含字符串“Hello”。考虑下面的代码,它要从这个Vector中删除“Hello”字符串:String s = "Hello";
int i = v.indexOf(s);
if(I != -1) v.remove(s);
String s = "Hello";
int i = v.indexOf(s);
if(I != -1) v.remove(i);
String s = "Hello"; v.remove(s);
最后,我们再来看一个有关Vector类的代码片段:for(int I=0; I ;I 如果v包含100,000个元素,这个代码片段将调用v.size()方法100,000次。虽然size方法是一个简单的方法,但它仍旧需要一次方法调用的开销,至少JVM需要为它配置以及清除堆栈环境。在这里,for循环内部的代码不会以任何方式修改Vector类型对象v的大小,因此上面的代码最好改写成下面这种形式:
int size = v.size(); for(int I=0; I ;I 虽然这是一个简单的改动,但它仍旧赢得了性能。毕竟,每一个CPU周期都是宝贵的。
拙劣的代码编写方式导致代码性能下降。但是,正如本文例子所显示的,我们只要采取一些简单的措施就能够显著地改善代码性能。