package juc.base;
/*1.字面量方式定义(例如String a = "abc")会检查字符串常量池中是否已经有这个字符串的引用,
如果有,就直接返回引用,没有,在堆中创建字符串对象对象再返回引用,这种情况下只会新建
一个字符串对象,并且这个字符串对象的引用已经存储在字符串常量池*/
/*2.new关键字定义(例如String a = new ("abc")),会在堆中创建一个字符串对象@111111(假如事先不存在),
并把引用存储在字符串常量池,并会在堆中再创建一个字符串变量@222222,注意,@111111和@222222虽然内容一样
但是他们时两个不同对象(内存地址不同),并且返回的时@222222这个引用*/
/*3.toString()方式(例如 String a = new StringBuilder("a").append("bc").toString()),
会创建4个对象,"a","bc"在堆中创建,并在字符串常量池中存储引用(假如事先在字符串常量池中不存在),一个StringBuilder对象在
堆中创建,一个"abc字"符串对象在堆中创建,特别注意,***此时"abc"对象的索引并不会在存储在字符串常量池中***
此时返回的时堆中的"abc"这个对象*/
/*4.intern()方法会先判断字符串常量池中是否有改字符串的引用,如果有,直接返回该引用,
如果没有,则存储该对象的引用,最后返回该对象的引用,注意,***字符串常量池只存储引用***
可以使用一个HashSet结构来理解,HashSet strSet = new HashSet<String>();*/
public class StringDemo {
public static void main(String[] args) {
// 情况1:返回true
String a = "张三";
System.out.println(a==a.intern());
// 情况1:返回false
String b = new String("李四");
System.out.println(b == b.intern());
// 情况3:返回true
String c = new StringBuilder("王").append("五").toString();
System.out.println(c==c.intern());
// 情况4:返回false
/*@@@@@@******这是一个大坑,看似和例子3几乎一模一样,但是需要注意的时,"java"这个字符串,
在JVM加载后就已经被存储到了字符串常量池,所以在执行d.intern()时实际不会
把原先堆中的那个"java"字符串对象的以用存储到字符串常量池中,所以2者并不相等*/
String d = new StringBuilder("ja").append("va").toString();
System.out.println(d==d.intern());
}
}