关于String如何创建对象

1.先简单看一看JVM内存结构


方法区: 该区为各个线程共享,用于存储已经被虚拟机加载的类信息、常量、静态变量、即时编译器编译出来的代码等数据。
常量池就在这个区域

堆: Heap区被所有的线程共享,在虚拟机启动时创建。此区的功能就是存放对象实例,几乎所有的对象实例都是在这里分配内容。Heap区垃圾回收器管理的主要区域。

2.创建字符串对象

public class TestDemo {
    public static void main(String[] args) {
        String a = "1";
        String b = "1";
        String c = b;
        System.out.println(a == b);
        System.out.println(c == b);

        String d = new String("test");

        String e = new String("1");
        String f = new String(a);
        System.out.println(d == a);
        System.out.println(e == d);
        
        String g = "hello" + "tomorrow";
        String h = new String("hello") + new String("world");
    }
}

首先,看一下

        String a = "1";
        String b = "1";
        String c = b;
        System.out.println(a == b);
        System.out.println(c == b);


控制台返回 true

分析上述代码的输出结果: 上述代码,只创建了一个对象

  • 首先,jvm在编译阶段会判断方法区常量池中是否有 “1” 这个常量对象(String a = "1";
    如果有,a直接指向这个常量的引用
    如果没有,就在常量池里创建这个常量对象
  • 此过程并没有在堆中创建对象
  • String b = "1"; 直接将 常量对象"1" 的地址交给了b,String c = b; 将 b 指向的 常量对象"1" 的地址交给了c
  • 当使用==判断时,都是在对比 常量池中的 常量对象"1" 的地址,故而相同,返回true

接着来看看

        String d = new String("test");

分析: 上述代码,创建了两个对象

  • 首先,jvm在编译阶段会判断方法区常量池中是否有 “test” 这个常量对象,没有就创建
  • 其次,通过 new 在 堆 中创建 String对象,d 指向的就是这个String对象的地址

继续看

        String e = new String("1");
        String f = new String(a);
        System.out.println(d == a);
        System.out.println(e == d);
        System.out.println(e == f);


控制台返回 false

分析上述代码的输出结果: 上述代码,创建了2个对象

  • 首先,常量池中已经有了 “1” ,且 a 指向的也是 “1” 的地址
  • 所以,此过程只在堆中用new创建两个 String 对象
  • 虽然他们的字符串常量值都是 1,但是 e和f 指向的是两个不同的String对象的地址,所以返回值都为false

最后看:

        String g = "hello" + "tomorrow";
        String h = new String("hello") + new String("world");

首先来分析String g = "hello" + "tomorrow"; 只创建了1个对象

  • jvm编译阶段过编译器优化后会把字符串常量直接合并成"hellotomorrow",所以最终只在常量池中创建了一个 “hellotomorrow” 常量对象

接着来看String h = new String("hello") + new String("world"); 创建了6个对象

  • 首先 new 创建了一个 StringBuilder() 对象
  • 接着 h = new String("hello") + new String("world") 创建了 4 个对象(和上述创建过程相同)
  • append()之后是StringBuilder对象,要转换成String对象,需要toString()toString()会创建一个新的String对象,这里创建了1个对象

3.补充

        Integer m = 3;//在常量池中创建一个常量对象 "3"
        String s = m.toString(m);

注意: 此过程调用Object.toString(),并没有在常量池中创建新的对象。

  • 1
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 3
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

364.99°

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值