想必大部分学过java的人都应该做过这种题目:
1 public class Test {
2 public static void main(String[] args) {
3 //第一个字符串
4 String s1="hello";
5
6 //第二个字符串
7 String s2="hello";
8
9 //比较s1和s2是否相同
10 System.out.println(s1==s2);
11
12
13 /**
14 * 修改变量
15 */
16
17 s1=new String("hello");
18
19 s2=new String("hello");
20
21 //再次比较s1和s2是否相同
22 System.out.println(s1==s2);
23 }
24 };
最后的结果如图:
图一
对于这个结果,我相信没有人有异议!因为第一组是依靠常量池的结果,即String s1=“hello”时实际会在内存中栈的常量池中创建一个字符串常量“hello”对象,当String s2=“hello”时,因为常量池中已经存在,所以s2会直接指向同一个”hello“对象,也就是s1和s2对象的地址值是一致的,此时的String就有点类似与基本数据类型,所以用“==”来比较的时候,值为“true”。
而如果String s1=new String("hello")时,s1在堆中开辟出新的内存空间存放“hello”值,同理,s2也会在堆中重新开辟空间存放“hello”值,s1和s2所指向的值的地址不同,用“==”来比较的时候自然是不同的,值为“false”。
但不知道有多少人做过下面这道题目呢?
修改一下上题中的变量类型为Integer。
1 package action;
2
3 public class Test {
4 public static void main(String[] args) {
5 //第一个数值
6 Integer s1=127;
7
8 //第二个数值
9 Integer s2=127;
10
11 //比较s1和s2是否相同
12 System.out.println(s1==s2);
13
14
15 /**
16 * 修改变量
17 */
18
19 s1=128;
20
21 s2=128;
22
23 //再次比较s1和s2是否相同
24 System.out.println(s1==s2);
25 }
26 };
你的答案会是什么呢?
结果图如下:
图二
是不是有点出乎你的意料呢?
从原理上来说,Integer和String类型都属于引用类型,那么最后的结果应该和String类型的比较一致,而且就算Integer类型的没有常量池技术那么这两次的结果也应该一致啊,毕竟只是相差1而已!
这就是我这篇博客要说的java常量池的小陷阱了!
java中的基本数据类型的包装类有6种实现了常量池技术,他们分别是:Byte , Short,Integer,Long ,Characher ,Boolean。Double和Float没有实现,佐证如下图:
图三
直接定义常量是会报错的(这个佐证其实有点不靠谱,如果有谁知道,麻烦告诉一声,谢谢!)
这5种包装类在常量池的实现上与String有点区别,以Integer为例,当有Integer i=127时,实际上发生了如下操作:
Integer i=Integer.valueOf(127)
Integer.valueOf()方法基于减少对象创建次数和节省内存的考虑,缓存了[-128,127]之间的数字。此数字范围内传参则直接返回缓存中的对象。在此之外,直接new出来.
而在上述所说的6种有常量池技术的包装类中,除了Boolean以外,其他的值都不能超过 127 (小于或者等于) ,超过127常量池就不予自动创建对象,这时包装类就和普通的引用类型没有区别了。