美文网首页IT面试
019:使用String的conact()方法和“+”做字符串拼

019:使用String的conact()方法和“+”做字符串拼

作者: 程序熊大 | 来源:发表于2019-03-06 21:11 被阅读208次
    pexels-photo-1181298.jpeg

    参考答案

    1. concat()方法只接受字符串类型的参数,参数不能为空;
    2. concat()底层是依靠Arrays.copy()方法实现的
    3. 如果有必要的话,"+"会对参与连接的变量进行toString();
    4. ”+“底层是依靠StringBuilder实现的,Java编译器在字节码层面做了优化;

    知识点梳理

    先看个例子,代码如下:

    public class StringConcatExample2 {
    
        public static void main(String[] args) {
            System.out.println(new StringConcatExample2().testConcat("abc", null));
            System.out.println(new StringConcatExample2().testConcat2("abc", null));
    
    //        System.out.println(new StringContactExample2().testContact("abc", "ddd"));
    //        System.out.println(new StringContactExample2().testContact2("abc", "ddd"));
        }
    
        public String testConcat(String a, String b) {
            return a += b;
        }
    
        public String testConcat2(String a, String b) {
            return a.concat(b);
        }
    }
    

    这个例子的运行结果如下:


    屏幕快照 2019-03-07 上午10.28.38.png

    可以通过查看字节码和JDK源码来比较二者的不同,将上面的代码使用javac StringConcatExample2.java编译,然后使用javap -c StringConcatExample2,可以看到对应的字节码内容。

    testConcat()方法字节码如下所示,从第0行可以看出,编译器做了优化,运算符重载“+”在字节码层面生成了一个StringBuilder对象,然后依靠append()方法进行连接。

     public java.lang.String testConcat(java.lang.String, java.lang.String);
        Code:
           0: new           #10                 // class java/lang/StringBuilder
           3: dup
           4: invokespecial #11                 // Method java/lang/StringBuilder."<init>":()V
           7: aload_1
           8: invokevirtual #12                 // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;
          11: aload_2
          12: invokevirtual #12                 // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;
          15: invokevirtual #13                 // Method java/lang/StringBuilder.toString:()Ljava/lang/String;
          18: dup
          19: astore_1
          20: areturn
    

    testConcat2的字节码如下所示:

      public java.lang.String testConcat2(java.lang.String, java.lang.String);
        Code:
           0: aload_1
           1: aload_2
           2: invokevirtual #14                 // Method java/lang/String.concat:(Ljava/lang/String;)Ljava/lang/String;
           5: areturn
    

    concat()方法的源码实现如下所示,可以看出是依赖Arrays.copy方法来进行数据的移动。

    public String concat(String str) {
        int otherLen = str.length(); //这里可以看出,如果str未null,则会直接报NPE
        if (otherLen == 0) {
            return this;
        }
        int len = value.length;
        char buf[] = Arrays.copyOf(value, len + otherLen);
        str.getChars(buf, len);
        return new String(buf, true);
    }
    

    参考资料

    1. https://stackoverflow.com/questions/47605/string-concatenation-concat-vs-operator

    本号专注于后端技术、JVM问题排查和优化、Java面试题、个人成长和自我管理等主题,为读者提供一线开发者的工作和成长经验,期待你能在这里有所收获。 javaadu

    相关文章

      网友评论

        本文标题:019:使用String的conact()方法和“+”做字符串拼

        本文链接:https://www.haomeiwen.com/subject/thhfpqtx.html