一、给一个字符串变量赋值有两种方式

1. String str1=new String("This is a string");
2. String str2 ="This is a string";

第一种方式通过关键字new定义:编译程序先在字符串常量池查找,是否存在"This is a string"常量,如果不存在,则在字符串常量池开辟一个内存空间,存放"This is a string";如果存在,则不另外开辟空间,保证字符串常量区只有一个"This is a string",节省空间。然后在堆区,开辟一个空间,存放new出来的String对象,并在栈区开辟空间,存放变量名称str1,str1指向堆区new出来的String对象。

第二种方式直接定义:在字符串常量区查找是否存在"This is a string"常量,如果不存在,则在字符串常量区开辟一个内存空间,存放"This is a string";如果存在,则不另外开辟空间;在栈区开辟空间,存放变量名称str2,str2指向字符串常量池"This is a string"的内存地址。

java 字符串 存放地方 java字符串放在哪里_jvm

二、String的intern方法使用

      当调用intern()方法时,如果字符串常量池中存在该字符串,则返回池中该字符串的引用;否则,将此字符串添加到字符串常量池,并返回字符串的引用。

        jdk1.6与jdk1.7处理字符串的相同点:

        如果字符串常量池中存在,则返回池中字符串的引用。

        jdk1.6与jdk1.7处理字符串的不同点:

        如果字符串常量池中不存在该字符串,jdk1.6会将该字符串拷贝到字符串常量池中,而jdk1.7会在字符串常量池中生成该字符串实例的引用。

String str1=new StringBuilder("this is ").append("a string").toString();
//执行完这行代码后,常量池中会有"this is "和"a string",但是不会有"this is a string"。
System.out.println(str1.intern()==str1);//true
String str2=new String("this is a string"); 
//执行完这行代码后,常量池中会有"this is a string"。
System.out.println(str2==str2.intern());//false

三、字符串常量池在JVM数据区的演进

java 字符串 存放地方 java字符串放在哪里_jvm_02

这里顺带提下常量池与运行时常量池的区别:

常量池:也叫静态常量池,在编译后的class文件中,分为字面量和符号引用量,还没有加载到虚拟机JVM中;

运行时常量池:JVM加载class文件,并为静态常量池分配内存空间,此时的常量池即为运行时常量池。除了编译后产生的常量,这个常量池在运行时,还会增加新的常量:intern()方法生成的新的字符串常量、由基础数据生成的包装类型。