`
minijack
  • 浏览: 22696 次
  • 性别: Icon_minigender_1
  • 来自: 江苏泰州
最近访客 更多访客>>
社区版块
存档分类
最新评论

从一道诡异的程序说起-----对象池技术

阅读更多
这几天一直在看java面试题,其中有一个基本上一面都会问的问题。就是==与equal()的异同问题。这个问题本文不谈,本文谈谈在实验这==中遇到了一个问题。
程序如下:
        
 public class Test {
	public static void main(String[] args) {	 
		Integer i1 = 127;
		Integer i2 = 127;
		System.out.println(i1 == i2); 
		Integer i3 = 128;
		Integer i4 = 128;
		System.out.println(i3 == i4);		 
	         }
          }
一个月前的我毫无疑问,会得出这样一个答案: true true
因为127等于127,128等于128。 呵呵  很菜
然而现在的我会得出:false false
因为我懂了==比较的是对象引用,而不是内容的相等。
然而,经过当我把这些代码敲进运行时,发生了意想不到的事。true false
这个就比较诡异了。 同等情况下不可能会出现这种情况的。可是结果就是这样,没有办法肯定是有问题的。经过百度了一阵子终于解决了心中的诱惑,虽然具体细节不是很了解,但是能够明白点了。 其实这些都是一个叫:  对象池   的技术在搞鬼。

具体的讲解请看下文: java中的对象池技术

    java中的对象池技术,是为了方便快捷地创建某些对象而出现的,当需要一个对象时,就可以从池中取一个出来(如果池中没有则创建一个),则在需要重复重复创建相等变量时节省了很多时间。对象池其实也就是一个内存空间,不同于使用new关键字创建的对象所在的堆空间。本文只从java使用者的角度来探讨java对象池技术,并不涉及对象池的原理及实现方法。个人认为,如果是真的专注java,就必须对这些细节方面有一定的了解。但知道它的原理和具体的实现方法则不是必须的。

1,对象池中对象和堆中的对象Java代码 
public class Test{Integer i1=new Integer(1);   
   Integer i2=new Integer(1);   
//i1,i2分别位于堆中不同的内存空间   System.out.println(i1==i2);//输出false   
   Integer i3=1;   
   Integer i4=1;   
//i3,i4指向对象池中同一个内存空间      
System.out.println(i3==i4);//输出true//很显然,   
i1,i3位于不同的内存空间   
System.out.println(i1==i3);   
//输出false}  

public class Test{Integer i1=new Integer(1);
   Integer i2=new Integer(1);
//i1,i2分别位于堆中不同的内存空间   System.out.println(i1==i2);//输出false
   Integer i3=1;
   Integer i4=1;
//i3,i4指向对象池中同一个内存空间   
System.out.println(i3==i4);//输出true//很显然,
i1,i3位于不同的内存空间
System.out.println(i1==i3);
//输出false}


2,8种基本类型的包装类和对象池java中基本类型的包装类的大部分都实现了对象池技术,这些类是Byte,Short,Integer,Long,Character,Boolean,另外两种浮点数类型的包装类则没有实现。另外Byte,Short,Integer,Long,Character这5种整型的包装类也只是在对应值小于等于127时才可使用对象池,也即对象不负责创建和管理大于127的这些类的对象。
以下是一些对应的测试代码:
Java代码 
public class Test{   
public static void main(String[] args){      
//5种整形的包装类Byte,Short,Integer,Long,Character的对象,   //在值小于127时可以使用对象池      
Integer i1=127;      
Integer i2=127;      
System.out.println(i1==i2)//输出true   //值大于127时,不会从对象池中取对象      
Integer i3=128;      
Integer i4=128;      
System.out.println(i3==i4)//输出false   //Boolean类也实现了对象池技术   Boolean bool1=true;     
Boolean bool2=true;      
System.out.println(bool1==bool2);//输出true   //浮点类型的包装类没有实现对象池技术      
Double d1=1.0;      
Double d2=1.0;      
System.out.println(d1==d2)//输出false     
}   
}  


public class Test{
public static void main(String[] args){   
//5种整形的包装类Byte,Short,Integer,Long,Character的对象,   //在值小于127时可以使用对象池   
Integer i1=127;   
Integer i2=127;   
System.out.println(i1==i2)//输出true   //值大于127时,不会从对象池中取对象   
Integer i3=128;   
Integer i4=128;   
System.out.println(i3==i4)//输出false   //Boolean类也实现了对象池技术   Boolean bool1=true;  
Boolean bool2=true;   
System.out.println(bool1==bool2);//输出true   //浮点类型的包装类没有实现对象池技术   
Double d1=1.0;   
Double d2=1.0;   
System.out.println(d1==d2)//输出false  
}
}

3,String也实现了对象池技术String类也是java中用得多的类,同样为了创建String对象的方便,也实现了对象池的技术,测试代码如下:
Java代码 
public class Test{   
public static void main(String[] args){   
//s1,s2分别位于堆中不同空间   
String s1=new String("hello");   
String s2=new String("hello");   
System.out.println(s1==s2)//输出false//s3,s4位于池中同一空间   
String s3="hello";   
String s4="hello";   
System.out.println(s3==s4);//输出true   
}   
}  


估计看完了这篇文章,大家就很明白了。

这是本来学java不到一个月以来收获最最大的一天,并不是因为我学到了多么牛逼的技术,而是因为我学会发现问题,解决问题的思路。 正所谓 :实践是检验真理的唯一标准。学习技术什么问题拿不定,来出来跑跑就明白了。
分享到:
评论
4 楼 minijack 2011-05-11  
kaka-pan 写道
富哥啊  有点问题!!!!
Integer i1=127;     
Integer i2=127;
这其实在jdk1.5以前版本是编译都不能通过的,1.5之后自动装箱机制变成了
Integer i1 = new Integer(127);
Integer是int封装类,编译的时候其实变成了Integer i1 = Integer.valueOf(127),你可以看看jdk里面Integer源代码中的valueOf()方法:
public static Integer valueOf(int i) { 
    final int offset = 128; 
    if (i >= -128 && i <= 127) { 
      return IntegerCache.cache[i + offset]; 
    } 
    return new Integer(i); 
  } 

结论:
当这个值在-128和127之间时,会用缓存保存起来,供多次使用,以节约内存。
如果不在这个范围内,则创建一个新的Integer对象。即另外开辟一块内存,这个时候i1和i2将不再指向同一个对象。




public class Test{      
public static void main(String[] args){      
//s1,s2分别位于堆中不同空间      
String s1=new String("hello");      
String s2=new String("hello");      
System.out.println(s1==s2)//输出false//s3,s4位于池中同一空间      
String s3="hello";      
String s4="hello";      
System.out.println(s3==s4);//输出true      
}      
}    


上面这段代码如何解释???
3 楼 minijack 2011-05-11  
kaka-pan 写道
富哥啊  有点问题!!!!
Integer i1=127;     
Integer i2=127;
这其实在jdk1.5以前版本是编译都不能通过的,1.5之后自动装箱机制变成了
Integer i1 = new Integer(127);
Integer是int封装类,编译的时候其实变成了Integer i1 = Integer.valueOf(127),你可以看看jdk里面Integer源代码中的valueOf()方法:
public static Integer valueOf(int i) { 
    final int offset = 128; 
    if (i >= -128 && i <= 127) { 
      return IntegerCache.cache[i + offset]; 
    } 
    return new Integer(i); 
  } 

结论:
当这个值在-128和127之间时,会用缓存保存起来,供多次使用,以节约内存。
如果不在这个范围内,则创建一个新的Integer对象。即另外开辟一块内存,这个时候i1和i2将不再指向同一个对象。


恩恩  不错
2 楼 wupuyuan 2011-03-28  
这个平时还真没注意过,呵呵,又学习到了新东西!
1 楼 kaka-pan 2011-03-27  
富哥啊  有点问题!!!!
Integer i1=127;     
Integer i2=127;
这其实在jdk1.5以前版本是编译都不能通过的,1.5之后自动装箱机制变成了
Integer i1 = new Integer(127);
Integer是int封装类,编译的时候其实变成了Integer i1 = Integer.valueOf(127),你可以看看jdk里面Integer源代码中的valueOf()方法:
public static Integer valueOf(int i) { 
    final int offset = 128; 
    if (i >= -128 && i <= 127) { 
      return IntegerCache.cache[i + offset]; 
    } 
    return new Integer(i); 
  } 

结论:
当这个值在-128和127之间时,会用缓存保存起来,供多次使用,以节约内存。
如果不在这个范围内,则创建一个新的Integer对象。即另外开辟一块内存,这个时候i1和i2将不再指向同一个对象。

相关推荐

Global site tag (gtag.js) - Google Analytics