看《深入理解java虚拟机》时了解到方法区中存在运行时常量池(Runtime Constant Pool),运行时常量池会在类加载时载入class文件中的常量池信息(constant_pool table)。

public class Test{
    public static String a = "a";
    public static void main(){
        String b = "b";
    }
}

clipboard.png

字符串“a”和“b”在常量池表中的,理应应该进入运行时常量池。运行时常量池(绝大部分)是随着JVM运行时候,从常量池转化而来,每个Class对应一个运行时常量池。那么a,b不应该在每一个class类对应的运行时常量池内么?下面是试验代码

public class A
{
    static String a = "hello";
    static String b = "你好";
    public String c = "hello";
    public String d = "你好";
}

public class B {
    static String a = "hello";
    static String b = "你好";
    String c = "hello";
    String d = "你好";
}
public class Test {

    @org.junit.Test
    public void test(){
        System.out.println(A.a == B.a);
        System.out.println(A.b == B.b);

        A a = new A();
        B b = new B();
        System.out.println(a.c == b.c);
        System.out.println(a.d == b.d);
        System.out.println(A.a == a.c);
        System.out.println(A.b == a.d);
        String c = new String("hello");
        System.out.println(c == A.a);
        System.out.println(c.intern() == A.a);
    }
}

实验结果(JDK8)

true
true
true
true
true
true
false
true

说明字符串“a”和“b”都是一块内存地址。关于常量池还有一个概念就是字符串常量池(String Pool)。String Pool是JVM实例全局共享的,全局只有一个。所以“a”和“b”这两个字符串应该是在字符串常量池内,全局共享。

以上是我一直以来的疑惑,字符串常量池和运行时常量池到底是什么关系?查了很多资料都是没有说到。运行时常量池是在方法区内,同时《深入理解java虚拟机》说到

在目前已经发布的JDK1.7的HotSpot中,已经把原本放在永久带的字符串常量池移除

所以个人觉得字符串常量池应该是在堆中表示常量的常量池,而运行时常量池是在方法区中。


tao不是哭脸
119 声望3 粉丝