今天看到几个面试题问:String s=new String("abc")创建了几个String Object对象?对于这个问题有的说是一个,有的说是两个,在网上也查看了很多别人写的博客都不清楚,今天就让我来分析一下这个问题.
对于String s=new String("abc"),这句话我们来大致分析一下步骤:
1.在虚拟机栈中为String类型的s分配内存
2.在堆中为分配一块内存用来保存"abc"
3.将堆中指向"abc"的地址赋值给s
这样来分析的话是在堆中创建了一个对象,而有些人会认为s也是对象啊,这不是两个吗?其实不然,s只是一个对象的引用,他指向了在堆中创建的那个对象.
我们来分析new String("abc"),打开String源码的构造器来看:
public String(String original) {
this.value = original.value;
this.hash = original.hash;
}
我们会发现传入的是一个String类型,而我们传入的是一个"abc",所以可以看作是这样的:
String temp="abc";
String s=new String(temp)
对于使用String temp="abc",会在变成成Class文件时,进入Class文件的常量池中,看下图:
Class文件的常量池保存是用于存放编译器生成的各种字面量(Literal)和符号引用(Symbolic References)。 字面量就是我们所说的常量概念,如文本字符串、被声明为final的常量值等。
符号引用就是可以用来表示堆内存中一个地址,在编译阶段只是把java类编译成JVM能够识别的Class文件,此时还没有进入内存,也就没法指出一个类所在内存的地址,这时JVM会使用类的全限定名称来唯一的指明一个类,这样在被加载到JVM中就会根据这个全限定名称去查找这个类存在的内存,以前关于符号引用和直接引用比较迷,现在记录一下。
在Class文件被类加载器加载到虚拟机的时候,“abc”这个字符串就会被检测并放到字符串常量池中,其过程是首先判断字符串常量池中是否存在该字符串,如果不存在就将“abc”放入到字符串中,然后返回其地址,如果存在的话,直接返回其地址。
String s="abc";
String s1="abc";
System.out.println(s==s1); //true
所以我们知道了,在String s=new String("abc")这个问题上要看上下文语境,
String s="abc";
String s1=new String("abc");
System.out.println(s==s1); //false
对于上面的代码String s1=new String("abc")就创建了一个对象,因为上面这段代码相当于
String s="abc";
String temp="abc";
String s1=new String(temp);
System.out.println(s==s1); //false
而在执行String temp="abc"时,会判断在方法区的字符串常量池中是否存在"abc"这个对象,发现存在了,就直接返回其地址,所以我们在回答的时候应该考虑上下文语境来回来.
如果在字符串常量池中不存在其代表的对象,就创建两个,如果存在就创建一个.
如果前面的你都已经理解,那么在思考下面几个问题;
String s = new String("abc") + "ab";//涉及到几个对象
String s = "abc" + "ab";//涉及到几个对象
可以在评论里面讨论上面的问题。
如想深入了解字符串常量保存的位置和Java类加载过程的可以查看:类加载过程、初始化和实例化阶段分析