1. 什么是常量池
常量池可以分为两类:静态常量池和运行时常量池
-
静态常量池:通常是指在编译时期被确定,并被保存在已经编译好的.class文件中的一些数据。编译好的class 文件中,从开头依次是4个字节的magic number,4个字节的存储版本号,然后就是存放常量的常量池,常量池的入口是constant_pool_count容量计数值。其中主要存放两大类常量字面量Literal和符号引用两 Symbolic Reference。
- 字面量:文本字符串,声明为final常量值
- 符号引用量:包括类和接口的全限定名,字段名称和描述符,方法名称和描述符
运行时常量池:此时JVM加载类class文件后,会将常量池的内容进入JVM的方法区中的运行时常量池进行存放。运行时常量池的特点在于:具有动态性,在运行期间可以将新的常量放入池中,比如String类的intern()方法。
2. 为什么需要常量池
- 常量池可以避免频繁创建和销毁对象而影响系统的性能,实现某些对象的共享,节省内存空间。
3. String类与常量池
- 注意s5,因为拼接的时候ss作为变量,作为新字符串,它是在堆中的,因此与s1地址是不同的
String s1 = "abcdefg";
String s2 = "abc"+"defg";
String s3 = new String("abcdefg");
String s4 = s2;
String ss ="defg";
String s5 = "abc"+ss;
System.out.println(s1 == s2); //true
System.out.println(s1 == s3); //false
System.out.println(s1 == s4); //ture
System.out.println(s1 == s5); //false
- String intern() 方法:查找常量池中是否存在equal的字符串,如果有的话返回该字符串的引用,如果没有则添加自己的字符串到常量池。
4. 基本类型包装类与常量池
- Byte,Short,Integer,Long,Character,Boolean中有实现常量池的技术,这些包装类默认创建了【-128,127】相应类型的缓存数据,如果超出此范围仍然会创建新的对象。注意基本类型的包装类是类对象,因此创建的基本类型包装类实例变量是引用变量,在缓存区范围内的是引用常量池地址,而在缓存区之外的引用的则是堆中对象地址。
Integer i1 = 10;
Integer i2 = 10;
Integer i3 = 300;
Integer i4 = 300;
System.out.println(i1==i2); // true
System.out.println(i3 ==i4);// false
- 基本类型与包装类的比较时会自动拆箱成相应基本类型,对值进行比较
int i =40;
Integer i5 = new Integer(40);
System.out.println(i==i5);//true