前言Java 字符串底层是如何存储的,如何提高性能的,今天就来好好了解一下。 字符串的存储结构Jvm 有专门的字符串常量池用于存放字符串,存放字符串的数据结构是HashTable。 HashTable的数据结构如下: 看个案例: public class StringDemo { public static void main(String[] args) { String a = "11"; String b = new String("11"); System.out.println("a的HashCode:" a.hashCode()); System.out.println("b的HashCode:" b.hashCode()); System.out.println("a==b :" (a==b)); // 比较的指针地址 System.out.println("a.equals(b) :" a.equals(b)); // 比较的是hashCode } } 运行结果: 通过案例我们来详细说明一下,Jvm如何创建一个String字符串的。 String 字符串会创建多少个Oop(Oop 是指Java 对象在Jvm中的存在形式)? String a = "11"; 我们可以通过idea 来看创建了多少个Oop。 调式字符串赋值后。 char[] 和 String 都加一了。 说明创建了两个Oop char[] 对应 TypeArrayKlass String 对应 TypeArrayOopDesc 画图说明字符串在Jvm 中的存在形式: String a = new String ("11"); -- 创建了3和Oop 再来看一下String 拼接的案例 public class StringDemo2 { public static void main(String[] args) { String s1 = "1"; String s2 = "2"; String s3 = s1 s2; String s4 = "12"; System.out.println("s3 == s4: " (s3 == s4)); } } 运行结果: 拼接字符串的实现原理用一句话完美解释: new StringBuilder(s1).append(s2).toString(); // toString 方法中有一个new String的逻辑。 并且拼接后的字符串是不放入常量池的。 看看toString 的源码 public String toString() { // Create a copy, don't share the array return new String(value, 0, count); } new String(value, 0, count) ,这种方式创建的String 不放入常量池。 那有什么方式可以将拼接的字符串放入常量池吗?答案是肯定的。 public class StringDemo2 { public static void main(String[] args) { String s1 = "1"; String s2 = "2"; String s3 = s1 s2; s3.intern(); // 将拼接的字符串放入常量池 String s4 = "12"; System.out.println("s3 == s4: " (s3 == s4)); } } intern 方法就是将拼接的字符串放入常量池。 再来看一个案例: public class StringDemo2 { public static void main(String[] args) { final String s1 = "1"; final String s2 = "2"; String s3 = s1 s2; String s4 = "12"; System.out.println("s3 == s4: " (s3 == s4)); } } 运行结果: 原因是s1,s2 都是final修饰,表示不会变,那么String s3 = s1 s2; 其实也不会变,所以和 s3 = “12” 等价。 总结String 字符串对应数据存放在字符串常量池。 拼接字符串实际就是StringBuilder 拼接。 final 修饰的情况。 intern 方法的作用是 将字符串加入常量池。 来源:https://www./content-1-878401.html |
|