java知识整理-从String开始

字符串拼接
"+"的实现 两个字符串"+"的底层实现是StringBuilder,通过javap -c命令反编译代码之后可以很清楚的看到

public class StringTest2 { public static void main(String[] args) { String a = "a"; String b = "b"; System.out.println(a + b); } }

javap -c StringTest2.class 结果输出
public class com.junhua.base.StringTest2 { public com.junhua.base.StringTest2(); Code: 0: aload_0 1: invokespecial #1// Method java/lang/Object."":()V 4: returnpublic static void main(java.lang.String[]); Code: 0: ldc#2// String a 2: astore_1 3: ldc#3// String b 5: astore_2 6: getstatic#4// Field java/lang/System.out:Ljava/io/PrintStream; 9: new#5// class java/lang/StringBuilder 12: dup 13: invokespecial #6// Method java/lang/StringBuilder."":()V 16: aload_1 17: invokevirtual #7// Method java/lang/StringBuilder.append:(Ljava/lang/String; )Ljava/lang/StringBuilder; 20: aload_2 21: invokevirtual #7// Method java/lang/StringBuilder.append:(Ljava/lang/String; )Ljava/lang/StringBuilder; 24: invokevirtual #8// Method java/lang/StringBuilder.toString:()Ljava/lang/String; 27: invokevirtual #9// Method java/io/PrintStream.println:(Ljava/lang/String; )V 30: return }

字面量相加的时候,在编译的时候会做优化,不会新建StringBuilder,而是直接将字符串拼接好,然后去新建String对象
String c = "a" + "b"; // 不会产生StringBuilder对象

concat字符串拼接 concat字符串拼接使用的是字符数组扩容,形成新的字符数组的方式实现
public String concat(String str) { int otherLen = str.length(); 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); }

字符串常量池
字符串常量池的功能类事于一个StringTable,是一个Hash表的实现,被所有的类共享。在jDK1.7以后,由于可以支持的字符串常量池的长度增加,放在方法区已经不太适合了,就挪到了堆中。同时,新增了常量池大小的设置
-XX:StringTableSize=66666
String s1 = "String"; String s2 = "String"; s1 == s2;

java首先会在缓冲区查找是否有"String"这个常量对象,有就直接将其地址赋给s1,没有就创建一个"String",然后将其赋给s1; 然后String s2 = "String";
java同样会在缓冲区中查找"String",这次能查找到了,因为s1创建了一个"String",所以会将其地址赋给s2,如此,s1和s2便有了相同的地址。
只有在新建字面量的时候才存在使用常量迟到概念,如果直接新建一个String对象,则是直接开辟一块内存,保持该字符串。
String a1 = new String("String");

intern 方法 如果常量池中有某个字符串对象了,就将该引用返回;否则将创建一个字符串对象放到常量池中,并返回这个引用。intern()方法的关键是始终是用的常量池中的字符串对象,而不是使用堆中的对象。
String b = new String("a").intern(); String a = new String("a"); System.out.println(a == b); //false String c = "a"; System.out.println(a == c); //true

String对象的创建 String a = new String("String"); 这个过程会在内存中创建两个对象,一个在heap中,一个在常量池中(如果常量池中已经有了该字符串,则不需要在常量池中创建)。a这个引用指向的是heap中的这个内存区域(对象)。
String b = "String"; 这个过程首先会去字符串常量池中查找,如果字符串常量池中有值,则直接指向字符串常量池中的地址,如果不存在,则在字符串常量池中新建一个该字符串,并指向该字符串所对应的内存区域。
String的不可变性
String对象是不可变的,一旦生成了,就无法直接修改,每次都会创建新的字符串对象。String内部的用一个char[]来保存,这个属性是一个私有的属性,并不对外提供修改的方法,进而也就无法实现对这个string对象的修改。
StringBuffer vs StringBuilder 字符串是不可变的,每次对字符串的操作都回生成新的对象,舍弃掉老的对象,会造成很多额外的开销。StringBuffer和StringBuilder就是为了解决这个问题而生的。不同点在于StringBuffer是线程安全的,StringBuilder是线程不安全的。append()方法都是采用字符串复制的方法,不同点在于StringBuffer在每个方法上都会加synchronized
【java知识整理-从String开始】StringBufferStringBuilder的核心方法append(String)
public AbstractStringBuilder append(String str) { if (str == null) return appendNull(); int len = str.length(); // 对原始数组长度进行扩容 ensureCapacityInternal(count + len); // 将str中的char复制到value这个char数组中 str.getChars(0, len, value, count); count += len; return this; }

    推荐阅读