1. String str=new String(“abc”); 这行代码究竟创建了几个String对象呢?(易错!)
    因为有new,所以堆中必然有一个对象。另外,如果常量池中已有"Hello World",则不创建,没有,则在常量池中创建。所以这句代码,究竟在内存中创建一个还是两个对象,视情况而定。
    解释
    首先在string池内找,找的到,不创建string对象,否则创建,这样就一个string对象 ;
    遇到new运算符号了,在内存上创建string对象,并将其返回给s,又一个对象 。
  2. String str=“abc”;
    毫无疑问,这行代码创建了一个String对象。
  3. String a=“abc”; String b=“abc”; 那这里呢?
    答案还是一个。
  4. String a=“ab”+“cd”;
    答案是三个。

在JAVA虚拟机(JVM)中存在着一个字符串池,其中保存着很多String对象,并且可以被共享使用,因此它提高了效率。由于String类是final的,它的值一经创建就不可改变,因此我们不用担心String对象共享而带来程序的混乱。字符串池由String类维护,我们可以调用intern()方法来访问字符串池。
我们再回头看看String a=“abc”;,这行代码被执行的时候,JAVA虚拟机首先在字符串池中查找是否已经存在了值为"abc"的这么一个对象,它的判断依据是String类equals(Object obj)方法的返回值。如果有,则不再创建新的对象,直接返回已存在对象的引用;如果没有,则先创建这个对象,然后把它加入到字符串池中,再将它的引用返回。因此,我们不难理解前面三个例子中头两个例子为什么是这个答案了。
只有使用引号包含文本的方式创建的String对象之间使用“+”连接产生的新对象才会被加入字符串池中。对于所有包含new方式新建对象(包括null)的“+”连接表达式,它所产生的新对象都不会被加入字符串池中,对此我们不再赘述。因此我们提倡大家用引号包含文本的方式来创建String对象以提高效率,实际上这也是我们在编程中常采用的。
(stack):主要保存基本类型(或者叫内置类型)(char、byte、short、int、long、float、double、boolean)和对象的引用,数据可以共享,速度仅次于寄存器(register),快于堆。
(heap):用于存储

对于String类对象来说它的对象值是不能修改的,也就是具有不变性。

看:
String s= "Hello ";
s= "Java ";
String s1= "Hello ";
String s2=new String( "Hello ");

但这里s="Java "; 为什么发生了改变呢?你别着急,让我告诉你说发生了什么事情:
在JVM工作过程中,会创建一片内存空间专门存入string对象。我们把这片内存空间叫做string池。

String s = "Hello "; 当JVM看到 "Hello ",在string池创建string对象存储它,并将他的引用返回给s。
s= "Java ",当JVM看到 "Java ",在string池创建新的string对象存储它,再把新建的string对象的引用返回给s。而原先的 "Hello "仍然在string池内。没有消失,他是不能被修改的。

所以我们仅仅是改变了s的引用,而没有改变他所引用的对象,因为string对象的值是不能被修改的。

String s1 = “Hello”; JVM首先在string池内里面看找不找得到字符串 “Hello”, 如果找得到,返回他的引用给s1,否则创建新的string对象,放到string池里。这里由于s = "Hello"了,对象已经被引用,所以依据规则s和s1都是引用同一个对象。所以ss1将返回true。(,对于非基本类型,是比较两引用是否引用内存中的同一个对象,比较得是引用得同一个对象得地址值)

String s2 = new String("Hello ");jvm首先在string池里面看找不找的到字符串 “Hello”,找到,不做任何事情,否则,创建新的string对象,放到string池里面。由于遇到了new,还会在内存上(不是string池里面)创建string对象存储 "Hello ",并将内存上的(不是string池内的)string对象返回给s2。所以s==s2将返回false,不是引用同一个对象。