zoukankan      html  css  js  c++  java
  • [改善Java代码]自由选择字符串拼接方法

    对一个字符串拼接有三种方法:加号,contact方法,StringBuffer或者StringBuilder的append方法,其中加号是最常用的.其他两种方式偶尔会出现在一些开源项目中,那么这三者有什么区别?

            //加号拼接
            str += "c";
            //concat方法连接
            str = str.concat("c");

    以上是两种不同的字符串拼接方式,循环5万次后再检查执行的时间,加号方式执行的时间是1438毫秒,而concat方法的执行时间是703毫秒,时间相差一倍,如果使用StringBuilder方式,执行时间会更少.

     1 public class Client {
     2     public static final int MAX_LOOP = 50000;
     3     public static void main(String[] args) {
     4         doWithPlus();
     5         doWithConcat();
     6         doWithStringBuffer();
     7         
     8         String str ="abc";
     9         String str1 = str.concat("1");
    10         String str2 = "abc1";
    11         System.out.println(str1 == str2);
    12     }
    13     
    14     public static void doWithPlus(){
    15         String str = "a";
    16         long start = System.currentTimeMillis();
    17         for(int i=0;i<MAX_LOOP;i++){
    18             str += "c";
    19             //str = new StringBuilder(prefix).append("c").toString();
    20         }
    21         long finish = System.currentTimeMillis();
    22         System.out.println("doWithPlus:" + (finish - start) + "ms");
    23     }
    24     
    25     public static void doWithConcat(){
    26         String str = "a";
    27         long start = System.currentTimeMillis();
    28         for(int i=0;i<MAX_LOOP;i++){
    29             str = str.concat("c");
    30         }
    31         long finish = System.currentTimeMillis();
    32         System.out.println("doWithConcat:" + (finish - start) + "ms");
    33     }
    34     
    35     public static void doWithStringBuffer(){
    36         StringBuilder sb = new StringBuilder("a");
    37         long start = System.currentTimeMillis();
    38         for(int i=0;i<MAX_LOOP;i++){
    39             sb.append("c");
    40         }
    41         String str = sb.toString();
    42         long finish = System.currentTimeMillis();
    43         System.out.println("doWithStringBuffer:" + (finish - start) + "ms");
    44     }
    45 }

    运行结果:

    doWithPlus:1559ms
    doWithConcat:748ms
    doWithStringBuffer:2ms
    false

    StringBuffer的append方法的执行时间是0毫秒.说明时间非常的短(毫秒不足以计时,可以使用纳秒进行计算).这个实验说明在字符串拼接的方式中,append方法最快,concat方法次之,加号最慢,这是为何呢?

    (1)"+"方法拼接字符串

    虽然编译器对字符串的加号做了优化,它会使用StringBuilder的append方法进行追加,按道理来说,其执行时间应该也是0毫秒,不过它最终是通过toString方法转换成String字符串的,例子中"+"拼接的代码与如下代码相同:

    str = new StringBuilder(str).append("c").toString();

    它与纯粹的使用StrignBuilder的append方法是不同的,意思每次循环都会创建一个StringBuilder对象,二是每次执行完毕都要调用toString方法将其转换为字符串------它的时间都耗费在这里了.

    (2)concat方法拼接字符串

     1     public String concat(String str) {
     2         int otherLen = str.length();
     3         //如果追加的字符串长度为0,着返回字符串本身
     4         if (otherLen == 0) {
     5             return this;
     6         }
     7         int len = value.length;
     8         char buf[] = Arrays.copyOf(value, len + otherLen);
     9         //追加的字符串转化成字符数组,添加到buf中
    10         str.getChars(buf, len);
    11         //复制字符数组,产生一个新的字符串
    12         return new String(buf, true);
    13     }

    其整体看上去就是一个数组的拷贝,虽然在内存中的处理都是原子性操作,速度非常快,不过,注意看最后的return语句,每次的concat操作都会新创建一个String对象,这就是concat速度慢下来的真正原因,它创建了5万个String对象.

    (3)append方法拼接字符串

    StringBuilder的append方法直接由父类AbstractStringBuilder实现,其代码如下....

        public AbstractStringBuilder append(String str) {
            if (str == null) str = "null";//如果是null值,则把null作为字符串处理
            int len = str.length();
            ensureCapacityInternal(count + len);//加长,并作数组拷贝
            str.getChars(0, len, value, count);
            count += len;
            return this;
        }

    整个append方法都在做字符组处理,加长,然后数组拷贝,这些都是基本数据处理,没有新建任何对象,所以速度也就最快了.

    例子中是在最后通过StringBuffer的toString返回了一个字符串,也就是在5万次循环结束之后才生成了一个String对象.

    "+"非常符合我们的编码习惯,适合人类阅读,在大多数情况下都可以使用加号操作,只有在系统性能临界的时候才考虑使用concat或者apped方法.

    而且很多时候,系统的80%的系能消耗是在20%的代码上,我们的精力应该更多的投入到算法和结构上.

  • 相关阅读:
    Visual C++ 打印编程技术-内存设备环境
    MySQL存储引擎
    记录阿里云服务器docker安装wordpress
    记录dockerfile参数
    记录一次 在公网使用FRP内网穿透开源软件,通过SSH连接内网服务器
    记录一次docker安装zabbix5.0
    记录一次zabbix邮件告警搭建过程和问题处理
    记录一次yum-config-manager命令的使用
    记录一次解决zabbix5.0图形化界面文字乱码的问题
    记录一次查看本地端口10050被哪个IP地址访问
  • 原文地址:https://www.cnblogs.com/DreamDrive/p/5660256.html
Copyright © 2011-2022 走看看