Java里String str = new String(“Hello“);会创建几个对象?

谈到这个问题首先得知道String的两种赋值方式:

    第一种是通过“字面量”赋值。 例如String str="hello";

    第二种是通过new关键字创建新对象。例如String str=new String("hello");

这两种方式到底有什么不同?程序执行的时候内存到底有几个实例?实例存在哪?字面量又存在了哪里?变量又存在哪?概念很容易搞混。下面我们一个个的讲。

虚拟机内存主要分为三块:

    堆:存放对象实例和数组。
    栈:存放基本类型,以及对象的引用。
    方法区:“类”被加载后的信息,常量、静态变量存放在这儿。

下面开始说String,比如下面这个main方法里声明一个字符串str

    package com.song.test;
     
    public class StringTest {
        public static void main(String[] args) {
            String str = new String("hello");
        }
    }

我们可以把

String str = new String("hello");

这行代码分成String str、=、"hello"和new String()四部分来看。String str只是名为str的String类型的变量,因此并没有创建对象;=是对变量str进行初始化,将某个对象的引用赋值给它,显然也没有创建对象;那么,new String("hello")为什么又能被看成"hello"和new String()呢?

我们来看一下被我们调用了的String的构造器:  

public String(String original) {  //other code ...  }   大家都知道,我们常用的创建一个类的实例(对象)的方法有以下两种:

一、使用new创建对象。

二、调用Class类的newInstance方法,利用反射机制创建对象。

我们正是使用new调用了String类的上面那个构造器方法创建了一个对象,并将它的引用赋值给了str变量。同时我们注意到,被调用的构造器方法接受的参数也是一个String对象,这个对象正是"hello"。由此我们又要引入另外一种创建String对象的方式的讨论——引号内包含文本。

这种方式是String特有的,并且它与new的方式存在很大区别。  

String str="hello";  
毫无疑问,这行代码创建了一个String对象“hello”。

String a="hello";  String b="hello"; 那这里呢?

答案还是一个。

String a="he"+"llo";   再看看这里呢?
答案是三个“he”、“llo”、“hello”

在JAVA虚拟机(JVM)中存在着一个字符串池,其中保存着很多String对象,并且可以被共享使用,因此它提高了效率。由于String类是final的,它的值一经创建就不可改变,因此我们不用担心String对象共享而带来程序的混乱。字符串池由String类维护,我们可以调用intern()方法来访问字符串池。  

我们再回头看看String a="hello";,这行代码被执行的时候,JAVA虚拟机首先在字符串池中查找是否已经存在了值为"hello"的这么一个对象,它的判断依据是String类equals(Object obj)方法的返回值。如果有,则不再创建新的对象,直接返回已存在对象的引用;如果没有,则先创建这个对象,然后把它加入到字符串池中,再将它的引用返回。因此,我们不难理解前面三个例子中头两个例子为什么是这个答案了。

只有使用引号包含文本的方式创建的String对象之间使用“+”连接产生的新对象才会被加入字符串池中。对于所有包含new方式新建对象(包括null)的“+”连接表达式,它所产生的新对象都不会被加入字符串池中,对此我们不再赘述。因此我们提倡大家用引号包含文本的方式来创建String对象以提高效率,实际上这也是我们在编程中常采用的。

因为new关键字会在堆申请一块新内存来创建对象。虽然字面还是“hello”,但是完全是不同的对象,有不同的内存地址。

字符串常量池在内存中什么位置呢?

    public class Test {  
      
        public static void main(String[] args) {  
            String str = "abc";  
            char[] array = {'a', 'b', 'c'};  
            String str2 = new String(array);  
            //使用intern()将str2字符串内容放入常量池  
            str2 = str2.intern();  
            //这个比较用来说明字符串字面常量和我们使用intern处理后的字符串是在同一个地方  
            System.out.println(str == str2);  
            //那好,下面我们就拼命的intern吧  
            ArrayList<String> list = new ArrayList<String>();  
            for (int i = 0; i < 10000000; i++) {  
                String temp = String.valueOf(i).intern();  
                list.add(temp);  
            }  
        }  
    }

执行一下,会怎么样?

true
Exception in thread "main" java.lang.OutOfMemoryError: PermGen space
        at java.lang.String.intern(Native Method)
        at Test.main(Test.java:16)
Java Result: 1

  异常信息告诉我们PermGen 满了。奥,我知道字符串常量池在哪了。PermGen就是jvm规范中所谓的方法区。

     这里偷懒了一下,只是指定了很大的数10000000让PermGen 溢出,不过时间可能长点。勤快的人还是自己指定java运行的内存比较好,稍小点就能验证。

 
————————————————
版权声明:本文为CSDN博主「songfelicity」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/songxinfeng1989/article/details/79576225

String s1 = new String("abc")到底创建了几个对象?底层原理是什么?_karry132的博客-CSDN博客

  • 1
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值