String s=new String("abc")创建了几个String Object对象?

        今天看到几个面试题问: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类加载过程的可以查看:类加载过程、初始化和实例化阶段分析

  • 13
    点赞
  • 17
    收藏
    觉得还不错? 一键收藏
  • 9
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 9
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值