使用String还是StringBuffer

 出于方便的考虑,我们在进行字符串的内容处理的时候往往会出现以下的代码:
        String result="";
        result+="ok";

这段代码看上去好像没有什么问题,但是需要指出的是其性能很低,原因是java中的String类不可变的(immutable),这段代码实际的工作过程会是如何的呢?通过使用javap工具我们可以知道其实上面的代码在编译成字节码的时候等同的源代码是:
        String result="";
        StringBuffer temp=new StringBuffer();
        temp.append(result);
        temp.append("ok");
        result=temp.toString();

短短的两个语句怎么呢变成这么多呢?问题的原因就在String类的不可变性上,而java程序为了方便简单的字符串使用方式对+操作符进行了重载,而这个重载的处理可能因此误导很多对java中String的使用。
下面给出一个完整的代码:
  1. public class Perf {
  2.   public static String detab1(String s)
  3.   {
  4.     if (s.indexOf('/t') == -1)
  5.       return s;
  6.     String res = "";
  7.     int len = s.length();
  8.     int pos = 0;
  9.     int i = 0;
  10.     for (; i < len && s.charAt(i) == '/t'; i++) 
  11.     {
  12.       res += "        ";
  13.       pos += 8;
  14.     }
  15.     for (; i < len; i++) 
  16.     {
  17.       char c = s.charAt(i);
  18.       if (c == '/t') {
  19.         do {
  20.           res += " ";
  21.           pos++;
  22.         } while (pos % 8 != 0);
  23.       }
  24.       else {
  25.         res += c;
  26.         pos++;
  27.       }
  28.     }
  29.     return res;
  30.   }
  31.       
  32.   public static String detab2(String s)
  33.   {
  34.     if (s.indexOf('/t') == -1)
  35.       return s;
  36.     StringBuffer sb = new StringBuffer();
  37.     int len = s.length();
  38.     int pos = 0;
  39.     int i = 0;
  40.     for (; i < len && s.charAt(i) == '/t'; i++) 
  41.     {
  42.       sb.append("        ");
  43.       pos += 8;
  44.     }
  45.     for (; i < len; i++) {
  46.       char c = s.charAt(i);
  47.       if (c == '/t') {
  48.         do {
  49.           sb.append(' ');
  50.           pos++;
  51.         } while (pos % 8 != 0);
  52.       }
  53.       else {
  54.         sb.append(c);
  55.         pos++;
  56.       }
  57.     }
  58.     return sb.toString();
  59.   }
  60.         
  61.   public static String testlist[] = {
  62.     "",
  63.     "/t",
  64.     "/t/t/tabc",
  65.     "abc/tdef",
  66.     "1234567/t8",
  67.     "12345678/t9",
  68.     "123456789/t"
  69.   };
  70.         
  71.   public static void main(String args[])
  72.   {
  73.     for (int i = 0; i < testlist.length; i++) {
  74.       String tc = testlist[i];
  75.       if (!detab1(tc).equals(detab2(tc)))
  76.         System.err.println(tc);
  77.      }
  78.         
  79.      String test_string =
  80.        "/t/tthis is a test/tof detabbing performance";
  81.      int N = 5000;
  82.      int i = 0;
  83.         
  84.      long ct = System.currentTimeMillis();
  85.      for (i = 1; i <= N; i++)
  86.        detab1(test_string);
  87.      long elapsed = System.currentTimeMillis() - ct;
  88.      System.out.println("String time = " + elapsed);
  89.          
  90.      ct = System.currentTimeMillis();
  91.      for (i = 1; i <= N; i++)
  92.        detab2(test_string);
  93.      elapsed = System.currentTimeMillis() - ct;
  94.      System.out.println("StringBuffer time = " 
  95.        + elapsed);
  96.    }
  97. }

执行以上代码的结果可以看到使用StringBuffer的版本的方法比使用String版本的一般都快十倍以上(本人使用的是JDK1.4.0),你可以执行一下看看结果到底如何。
因此得到的结论是: 如果你对字符串中的内容经常进行操作,特别是内容要修改时,那么使用StringBuffer,如果最后需要String,那么使用StringBuffer的toString()方法好了!也许这就是你的程序的性能瓶颈!
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值