常量池是什么
说到常量池,我们一般在JVM中把常量池分为运行时常量池和 字符串常量池。
运行时常量池:用于存放编译时期生成的各种字面量和符号引用。
字面量: 类似于我们平常说的常量,如String s = "“bb” , 字符串"bb"就是字面量。 还有就是final修饰的变量。
符号引用:类及其实现的接口的全限定名(如String类的全限定名就是java.lang.String)和变量的名称和类型,方法的名称和描述符(方法的参数类型+返回值类型)
字符串常量池:顾名思义,是储存字符串的。
常量池的位置
这两个常量池的位置是可以分3个阶段来描述的。
①在JDK1.7之前,如JDK1.6 : 字符串常量池是在运行时常量池里面用于储存文本字符串,而运行时常量池是在方法区(此时的实现是永久代)中。
②在JDK1.7的时候,字符串常量池被移出来,放于堆中,而运行时常量池依然在方法区中。
③在JDK1.7之后,如JDK1.8 : 用元空间的实现方式取代了永久代,字符串常量池仍然在堆中,运行时常量池在方法区(此时的实现方式是元空间)中。
intern方法介绍
String s = "22";
s.intern();
①在JDK1.7之前,当字符串调用intern方法时,虚拟机会去字符串常量池中查看是否有该对象,如果有,则直接返回该对象的地址。如果没有,则往字符串常量池中添加该对象,返回该对象的地址。
②在JDK1.7及其之后,当字符串调用intern方法,虚拟机会去字符串常量池中查看是否有与该对象内容相同的对象,如果有,直接返回该对象;如果没有,则会在堆中把这个对象的引用放入常量池中,返回该引用。
字符串常量池的例子
String s1 = new String("abc");
String s2 = "abc";
String s3 = s1.intern();
System.out.println(s1 == s2);
System.out.println(s2 == s3);
System.out.println(s1 == s3);
String s4 = new String("2”)+ new String("2");
String s5 = "22";
System.out.println(s4 == s5);
答案:在JDK1.6中: false true false
还有一种答案:在JDK1.7中: false false fasle
Java基本类型封装类常量池
如Byte ,Short ,Integer ,Long , Character,Boolean 都要自己的常量池。注意,但是Float和Double类型没有。该常量池在方法区中。
要处理基本类型的变量的比较,只需要记住以下几点:
①对象数值在-128-127才能进入常量池
②当==两边出现运算符(如±*/),会进行拆箱,此时基本类型比较的是数值。
③Long的equals方法会先判断括号里面是不是Long类型,如果不是,直接返回false。
④包装类重写了equals方法,equals方法比较的是它们的值。
⑤如果= = 两边出现基本类型(如int),则封装类会进行自动拆箱,拆完之后比较的是值。
例子
int i = 1;
Integer a = 1;
Integer b = 2;
Integer f = 1;
Integer c = 133;
Integer d = 133;
Integer h = 3;
Long e = 3L;
System.out.println(a == f);
System.out.println(c == d);
System.out.println(i == a);
System.out.println(a.equals(f));
System.out.println(e.equals(h);
答案:true , false , true , true , false