目录
最近在复习java的时候,发现了一些很小的基础知识点:
java的引用数据类型-附String字符串常量池简介(别再用new String了)!
为什么写这篇文章,其实是因为虽然自己平时开发中经常用到,但是却没有整理总结过这个问题的答案,导致一下子没罗列完全,所以特此记录一下,希望对有需要的小伙伴们有帮助:
在网上搜了一轮资料,发现对于引用数据类型,大家都各抒己见,有人说3种、4种、5种、6种都有,这里我就不下结论说到底有几种引用数据类型,不管有多少种都可以一言以蔽之,那就是
除了8种基本数据类型之外的都属于引用类型!
8种基本数据类型:byte、char、short、int、long、float、double、boolean
除了上面这8种,其他的都属于引用类型,也就是对象实例,例如以下这些:
class(类)、Annotation(注解)、Enum(枚举)、String(字符串)、
Interface(接口)、 数组 []
1、方法中定义的基本数据类型变量(局部变量),会存放在jvm内存结构的“栈”区域。
2、方法外定义的全局变量/成员变量(基本数据类型或引用数据类型),都会存放在jvm内存结构的“堆”区域。
3、方法中定义的引用类型对象的对象引用【str】和引用地址【hashcode】,会存放在jvm内存结构的“栈”区域、而对象【new String("hahaha")】和对象引用地址对应的内存空间则是在jvm内存结构的“堆”区域,如:
- public static void main(String[] args) {
- String str = new String("hahaha");
- }
堆中有一块内存叫做“String字符串常量池”,该常量池会存放字符串对象的字面量,所以在上面的例子中,String字符串常量池里面还会再存放一个字面量【"hahaha"】。
字符串常量池的作用:如果String对象不是通过new方式创建时,如:
String str2 = "heihei";
此时str2对象就是放在String字符串常量池中,包括其字面量对象【"heihei"】也同样在常量池中,此后再有任何通过非new方式创建的String对象且字面量也为【"heihei"】的,都会直接引用常量池中该对象的地址,这就是String的不可变性,可以提高内存空间利用率。
而在前面提到的str对象,则会存放在堆中,但不会放在字符串常量池中,不过它的字面量会放在常量池中,且拥有自己的引用地址,也就是相当于产生了2个对象,一个str对象,一个字面量对象,它们的引用地址是不同的,于是做==比较时会出现以下情况:
- public static void main(String[] args) {
- String str1 = new String("abc");
- String str2 = new String("abc");
- String str3 = "abc";
- String str4 = "abc";
-
- System.out.println(str1==str2);
- //false:new对象会在堆中开辟新空间,所以两者是不同的引用地址
-
- System.out.println(str2==str3);
- //false:str2是堆中非Stirng常量池的引用地址,str3是String常量池中的引用地址,所以不同
-
- System.out.println(str3==str4);
- //true:两者都是String常量池中的同一个字面量的引用地址
- }
此时就会发现原来new一个常量池中不存在的String对象时,会创建两个对象,一个是String对象,一个是字面量对象(再次new时,由于常量池已经存在了,则不会再次创建相同的字面量对象)。
可以发现,字符串常量池是为了:以后再通过非new方式创建String对象时,可以直接去常量池拿,如果存在则直接引用同一块内存空间,而不需要重新开辟内容空间,从而提高内存空间利用率。
所以强烈建议使用非new的方式来创建String对象!!!
其实你在IDEA用new String()的时候,就会发现,编译器会提示该方式已经out了: