1 什么是常量
用 final
修饰的成员变量表示常量,值一旦给定就无法改变。
final
修饰的变量有3种:静态变量,实例变量和局部变量,分别表示 3种类型的常量。
2 常量池
常量池即 常量共享的一种方案
。
5种包装类(Byte,Short,Integer,Character,Long),都实现了常量池技术。内存中缓存了 -128 ~ 127
范围内的数据,其中 Byte类型全部进了缓存
。这也是为什么当我们初始化的值比128小时,引用都是用的同一个的原因。
两种浮点数类型的包装类 Float,Double并没有实现常量池技术。
Integer i1 = 40;
Integer i2 = 40;
System.out.println(i1==i2);// true
******** **********
Double d1 = 1.2;
Double d2 = 1.2;
System.out.println(d1==d2);// false
3 常量池的优点
常量池是为了避免频繁的创建和销毁对象而影响系统的性能,其实现了对象的共享。
例如 字符串常量池,在
编译阶段
就把所有的字符串文字放到一个常量池中
- 节省内存空间:常量池中所有相同的字符串常量被合并,只占用一个空间
- 节省运行时间:比较字符串时==比 equals() 快。
对于两个引用变量,只用==判断引用是否相等,就可以判断值是否相等Java源程序的运行:
编译 ---> 链接 ---> 运行
编译成class文件
,运行
4 常量池分类详解
在平时用到的最多的就是字符串常量池,字符串常量池是运行时常量池的一部分。在 jdk 1.7 之后,运行时常量池和字符串常量池都从方法区移到堆中。
4.1 字符串常量池
字符串常量池是运行时常量池的一部分,因为它比较重要,所以单独讲
在 HotSpot中,通过
StringTable类
实现功能,StringTable是一个Hash表
,默认长度大小1099
,被所有类共享。字符串常量由字符组成,保存在 StringTable上面。
在 jdk1.6当中,StringTable的长度是固定1099,如果存放在StringTable中的字符串很多,造成hash冲突的几率很大,链表过长,当通过String.intern()查找String Pool时,性能就会降低。在 jdk1.7当中,StringTable的长度可以通过-XX:StringTableSize设置
存放内容
String.intern()主要是为了复用字符串常量
,节省内存空间。在 jdk1.6及以前的版本,存放的都是字符串常量,使用String str="string"
声明的字符串都存储在这,例如:String str = “abc”;在 jdk1.7之后,String.intern()发生变化,如果字符串常量池不存在这个String对象,但如果堆区存在这个对象,直接复制到字符串常量池
,否则还是要创建字符串,然后返回字符串对象的引用。因此除了字符串常量
,也可以存放堆中字符串常量的引用
在JDK 1.7 之后,字符串常量池从方法区转移到堆中
4.2 静态常量池(class常量池)
class文件中保存着类的相关信息(版本,类、字段、方法、接口等信息),除此之外,还有Class常量池,用来存放编译器产生的各种字面量(Literal)
和符号引用(Symbolic References)
,每个class文件都有一个class常量池。
字面量相当于Java语言层面常量的概念
如:文本字符串(String),声明为 final的 常量值等
符号引用属于编译原理方面的概念,包括如下三种类型的常量
类和接口的全限定名
字段名称和描述符
方法名称和描述符
4.3 运行时常量池
就是class常量池被加载到方法区之后的版本
。
区别:
字面量
可以通过 String.intern()动态添加
符号引用
解析为直接引用(类加载的解析阶段)当类加载到内存之中,jvm会把 class常量池的内存存放到运行时常量池,所以,
运行时常量池也是每个class都有的
运行时常量池相对于CLass文件常量池的另外一个重要特征是具备动态性,Java语言并不要求常量一定只有编译期才能产生,也就是并非预置入CLass文件中常量池的内容才能进入方法区运行时常量池,运行期间也可能将新的常量放入池中,这种特性被开发人员利用比较多的就是String类的intern()方法。
JDK1.7及之后版本的 JVM 已经将运行时常量池从方法区中移了出来,在 Java 堆(Heap)中开辟了一块区域存放运行时常量池.所以jdk1.7之后,运行时常量池和字符串常量池都从方法区移到堆中