java 的字符串常量缓冲区(字符串池,字符串常量池)。常量字符串的特点:不能改变。
===============================================================================================
Java中字符串对象创建有两种形式,一种为字面量形式,如String str = "abc";
1. 执行语句String str="abc";时。首先查看字符串池中是否存在字符串"abc",如果存在则直接将“abc”赋给str,如果不存在则先在 字 符串池中新建一个字符串"abc",然后再将其赋给str.
工作原理:JVM检测这个字面量,这里我们认为没有内容为droid的对象存在。JVM通过字符串常量池查找不到内容为droid的字符串对象存在,那么会创建这个字符串对象,然后将刚创建的对象的引用放入到字符串常量池中,并且将引用返回给变量str1。
重要的知识点:String str1 = "abc" 这样不会重新创建一个对象。而是同样JVM还是要检测这个字面量,JVM通过查找字符串常量池,发现内容为”droid”字符串对象存在,于是将已经存在的字符串对象的引用返回给变量str2。注意这里不会重新创建新的字符串对象。
验证是否为str1和str2是否指向同一对象,我们可以通过这段代码
代码如下: System.out.println(str1 == str2); 结果为true。
内存结构如图所示:
解析:在定义String str = "abc" 的时候,"abc"是常量存储数据共享区,定义时首先会到数据共享区中查找是否存在“abc"这样的字符串,如果存储在那么就直接把"abc"的地址赋值给str,如果数据共享区中不存在"abc",那么就去创建一个”abc",地址为0X99.
知识点扩展(常量字符串的不可改变性):
问题1:两个字符串相加之后还是保持同一个地址吗?答:不同。
例子:String str = "abc";//这个abc的字符串常量对象的地址一开始为0X99,str变量指向的地址就是0x99.
str = str + “de";//而相加之后 会形成de这个字符串常量对象,地址为0X98.
相加完之后abcde这个字符串常量的对象的地址为0X97,此时str变量指向的地址变成了该地址0x97
问题2:String str2 = "abc" ;// 一开始在数据共享区也就是常量池中创建一个字符串常量对象abc,假设地址是0x97
String str3 = "abc";// 首先先去常量池中查找有没有常量abc,发现有,则将变量str3的地址指向常量池中的abc的地址0x97,所以此时str2和str3两个地址一样。
System.out.println("str2 == str3"); //输出true
====================================================================================
Java中字符串对象创建有两种形式,另一种就是使用new这种标准的构造对象的方法,如String str1 = new String("abc");//创建了两个对象。
当我们使用了new来构造字符串对象的时候,不管字符串常量池中有没有相同内容的对象的引用,新的字符串对象都会创建。
解析:String str = "abc" 在数据共享区创建的是一个对象。
String str1 = new String("abc")在堆中创建一个字符串的对象,然后再到数据共享区中创建一个字符串常量的对象,然后把堆中的对象指向数据共享区中的对象。即首先是New String()这个对象,然后才到数据共享区中查找有没有abc这个对象,如果有就不用创建了,直接指向就好,否则再创建一个对象,最后在讲堆中的new String()对象指向数据共享区中的对象。
================================================
String是我们必须掌握的一个知识点。
先展示效果代码:
String str = new String("abc");
String str1 = "abc";
System.out.println(str == str1); //双等号比较的地址
System.out.println(str.equal(str1)); //比较的值
结果:
false
true
原因解析:Java运行环境有一个字符串池,由String类维护。