转载自:https://www.cnblogs.com/liaochong/p/javabase.html,建议看原文,格式更清晰
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
想必大部分学过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,Boolean,Characher。Double和Float没有实现,佐证如下图:
图三
直接定义常量是会报错的(这个佐证其实有点不靠谱,如果有谁知道,麻烦告诉一声,谢谢!)
这5种包装类在常量池的实现上与String有点区别,以Integer为例,当有Integer i=127时,实际上发生了如下操作:
Integer i=Integer.valueOf(127)
Integer.valueOf()方法基于减少对象创建次数和节省内存的考虑,缓存了[-128,127]之间的数字。此数字范围内传参则直接返回缓存中的对象。在此之外,直接new出来.
而在上述所说的6种有常量池技术的包装类中,除了Boolean以外,其他的值都不能超过 127 (小于或者等于) ,超过127常量池就不予自动创建对象,这时包装类就和普通的引用类型没有区别了。
关于常量池的详细介绍,可以借鉴这篇博客(我也是从中得到原因的):
http://www.cnblogs.com/qinqinmeiren/archive/2011/07/19/2151683.html
关于Integer.valueOf()可以参考这篇:
http://blog.csdn.net/randyjiawenjie/article/details/7603427
转载自:https://www.cnblogs.com/liaochong/p/javabase.html