- String 内存模型
-
String pubNmStr="abc"; 直接给string赋值,其值直接放在常量池中,变量直接指向对应常量池。这里需要重点说一下如果String 的值有修改 如:pubNmStr += "d";那么在常量池中会创建一个新的字符串abcd,而之前的字符串abc会依然保存在于常量池中,直至GC回收。
-
Student student1= new Student();student1.setName("abc"); 如此以对象的属性为String类型的,同样该属性直接引用常量池中已有的字符串,在赋值是常量池中不存在该字符串,同样会在常量池中创建后再直接指向它。
- Student student2=new Student();student2.setName(pubNmStr);
同2. 这里可以再次证明1中的String是为创建对象直接指向常量池中的字符串的。 -
String pubNmStrAddStr = "ab" + "c"; 该方式赋值在是将两个字符串直接相加,由于两个都直接是常量。在运行之前我们的编译器会将其先直接编译。编译后的class文件中我们看到的是如下效果。所以它也是直接指向常量池中的字符串。
-
String pubNmNewStrIntern= new String("abc").intern();
这个intern方法会跳过在堆内存中创建一个新的new String对象,转而直接指向常量池中的字符串。 -
String pubNmNewStr= new String("abc"); 该命令会先检查常量池中是否有相同的字符串,这里已有不在创建,如果没有会创建一个。接下来会运行到new关键字,JVM就会在系统中直接在创建一个String对象,该对象的值虽然直接指向常量池中的abc,但是赋值给pubNmNewStr的是new出来的String对象。所以pubNmNewStr与1中的pubNmStr不会相等。相当于是pubNmNewStr中间多了一个对象地址中转。如下如第6条箭头流向。
- 第7.8.9个箭头如下图是案例:
String pubNmNewStrAddNewStr= new String("ab") + new String("c");
可以看到它在内存中的流程图是先分别创建"ab" 和"c"对象。然后又builder合并为一个新对象然后再toString为pubNmNewStrAddNewStr对象。
以上7个String和其中2到6和1的对比结果分别如下,可以理解基本的string内存模型。