首先看一个例子,通过这个例子更能快速理解String常量池
1 public static void main(String[] args) { 2 3 String a = "ab"; 4 String bb = "b"; 5 String b = "a" + bb; 6 String c = "a" + "b"; 7 String d = b.intern(); 8 9 System.out.println(a == b); 10 System.out.println(a == c); 11 System.out.println(a == d); 12 }
运行结果如下:
1 false 2 true 3 true
这是为什么呢,String类的final修饰的,以字面量的形式创建String变量时,jvm会在编译期间就把该字面量(“ab”)放到字符串常量池中,由Java程序启动的时候就已经加载到内存中了。这个字符串常量池的特点就是有且只有一份相同的字面量,如果有其它相同的字面量,jvm则返回这个字面量的引用。当我们创建String对象采用字面量形式时,JVM首先会对这个字面量进行检查,如果常量池中存放有该字面量,则直接使用,所以a == c为ture,否则创建新的对象并将其引用放入常量池中,所以 a == b 为false。b.intern()方法能使一个位于堆中的字符串在运行期间动态地加入到字符串常量池中(字符串常量池的内容是程序启动的时候就已经加载好了),所以a == d为ture。
----String常量池----
1.常量池中的对象从哪里来的?
当我们创建String对象采用字面量形式时,JVM首先会对这个字面量进行检查,如果常量池中存放有该字面量,则直接使用,否则创建新的对象并将其引用放入常量池中。
当我们通过new来创建一个字符串时,JVM就不会去常量池中找,而是直接在堆中新建一个对象。如果我们想把new的对象也放在常量池中,可以调用方法inner();
2.常量池中存的是对象还是引用?
常量池中存的是对对象的引用,存储于JVM的方法区中,而且引用的对象存储于堆中。
3.常量池中的对象会被GC回收吗?
当常量池中的引用没有被任何变量引用时,就会被GC回收!