谈到这个问题首先得知道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博客

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

  1. 【Java深入理解】String str = “a“ + “b“ + “c“到底创建了几个对象?

    String str = "a" + "b" + "c"到底创建了几个对象?这是我们在讨论中最经常遇到的一个问题同时也是面试题.我们都知道在 ...

  2. String str =new String(“abc“)和 String str = “abc“的比较

    String是一个非常常用的类,应该深入的去了解String 如: String str =new String("abc") String str1 = "abc&qu ...

  3. String str=“i“与 String str=new String(“i”)一样吗?

    面试题汇总 String str="i"与 String str=new String("i")一样吗? 不一样 因为内存的分配方式不一样.String str ...

  4. 8.String str=“i“与 String str=new String(“i”)一样吗?

    不一样 因为内存的分配方式不一样.String str="i"的方式,Java 虚拟机会将其分配到常量池中:而 String str=new String("i" ...

  5. String str 与 String str=new String() 区别

    1.当使用String str="abc",这种方式时,先去内存的Heap中找是否存在"abc"这个字符串,若存在,则将地址引用.若不存在则创建. 2.当使用S ...

  6. String str = new String(abc)创建了几个对象?结合源码解析

    String str = new String("abc")创建了几个对象?结合源码解析 首先,我们看一下jdk源码: 1 /** 2 * Initializes a newly ...

  7. String str=Hello 与 String str=new String(“Hello”)一样吗?

    为什么会输出上边的结果呢,String x = "Hello" 的方式,Java 虚拟机会将其分配到常量池中,而常量池中没有重复的元素,比如当执行"Hello" ...

  8. Java里的容器存放的元素必须是1个对象.

    我们经常看到这个句话: Java里存放的容器只能是1个对象. 一. 值类型和对象类型. 实际上, java里的变量可以分为两种类型, 一种是值类型. 一种是对象类型. 1.1 值类型变量 所谓值类型的 ...

  9. java 反射创建对象并赋值_java使用反射创建并操作对象的方法

    Class 对象可以获得该类里的方法(由 Method 对象表示).构造器(由 Constructor 对象表示).成员变量(由 Field 对象表示),这三个类都位于 java.lang.refle ...

  10. java基础进阶一:String源码和String常量池

    作者:NiceCui 本文谢绝转载,如需转载需征得作者本人同意,谢谢. 本文链接:http://www.cnblogs.com/NiceCui/p/8046564.html 邮箱:moyi@moyib ...

最新文章

  1. 你知道什么是数据科学?如何把数据变成产品么?
  2. 转载:asp.net生成缩略图通用函数(支持多种生成方式)
  3. 【Android 逆向】类加载器 ClassLoader ( 启动类加载器 | 扩展类加载器 | 应用类加载器 | 类加载的双亲委托机制 )
  4. git怎么上传文件到别人的仓库_Git将本地代码上传至远程仓库
  5. angular js一factory,service,provider创建服务
  6. pysql与mysql的区别_postgresql与mysql的区别是什么
  7. RTL行为级仿真、综合后门级功能仿真和时序仿真
  8. 最全整理浏览器兼容性问题与解决方案(转)
  9. 当年,兔子学姐靠这个面试小抄拿了个22k
  10. Linux zip加密压缩
  11. 如何将Spring Bean注入到JSF Converter
  12. Java 面试 ——可变参数、初始化数据块、设计秒杀系统
  13. 算法:求数的幂次方powx-n
  14. git 下载指定历史版本
  15. python实现爬取网页将特定信息存入excel
  16. 创新工场投资经理:创业就是九死一生
  17. 领存!专注为军工业提供计算、存储、加密以及数据安全的系统服务
  18. vue项目创建之后 ESLint导致第一行毛毛虫
  19. corn表达式的使用
  20. IDEA3.5最新版激活码

热门文章

  1. Python 有那么神吗?
  2. thinkPHP 空模块和空操作、前置操作和后置操作 详细介绍(十四)
  3. 按位与、或、异或等运算方法(转)
  4. 分页内存与非分页内存导致的蓝屏死机问题
  5. 在电子政务项目中如何应付长官意识和拖沓的工作作风?
  6. 编译问题处理:undefined symbol: OPENSSL_init_crypto
  7. DPDK-UIO 驱动认识
  8. Win10环境下使用VS2015编译PJProject
  9. vuejs2.0从入门到放弃--入门实例(四)
  10. freeswith 录制 MP4格式视频