1、String s = new String(“aa”); 创建几个对象?
String s = new String("aa");
- 此时创建了2个对象! 使用 new 关键字创建一个字符串对象时,Java 虚拟机会先在字符串常量池中查找有没有‘aa’这个字符串对象。
- 如果有,就不会在字符串常量池中创建‘aa’这个对象了,直接在堆中创建一个‘aa’的字符串对象,然后将堆中这个‘aa’的对象地址返回赋值给变量s。
- 如果没有,先在字符串常量池中创建一个‘aa’的字符串对象,然后再在堆中创建一个‘aa’的字符串对象,然后将堆中这个‘aa’的字符串对象地址返回赋值给变量 s。
2、为什么要先在字符串常量池中创建对象,然后再在堆上创建呢?这样不就多此一举了?
由于字符串的使用频率实在是太高了,所以 Java 虚拟机为了提高性能和减少内存开销,在创建字符串对象的时候进行了一些优化,特意为字符串开辟了一个字符串常量池。
通常情况下,我们会采用双引号的方式来创建字符串对象,而不是通过 new 关键字的方式:
String s = "bb";
- 当执行 String s = “bb” 时,Java 虚拟机会先在字符串常量池中查找有没有“bb”这个字符串对象。
- 如果有,则不创建任何对象,直接将字符串常量池中这个“bb”的对象地址返回,赋给变量 s;
- 如果没有,在字符串常量池中创建“bb”这个对象,然后将其地址返回,赋给变量 s。
- 有了字符串常量池,就可以通过双引号的方式直接创建字符串对象,不用再通过 new 的方式在堆中创建对象了。
- new 的方式始终会创建一个对象,不管字符串的内容是否已经存在,而双引号的方式会重复利用字符串常量池中已经存在的对象。
String s = new String("aa");
String s1 = new String("aa");
按照我们之前的分析,这两行代码会创建三个对象,字符串常量池中一个"aa",堆上两个,s、s1。
String s = "bb";
String s1 = "bb";
这两行代码只会创建一个对象,就是字符串常量池中的那个"bb"。这样的话,性能肯定就提高了;