天天看点

java内存分配问题

java的内存分配分为两种基本数据类型。和对象类型。对8类基本数据类型,直接在栈上分配。对于对象既在栈上分配,也在堆上分配,在栈上分配引用的地址,在堆上分配new出来的对象。通过栈上分配的地址找到堆上new出来的对象。这些基础知识基本上都很简单,其实通过java的内存分配。我们可以很清楚的理解java里面非常著名的垃圾回收机制。java的垃圾回收机制,只回收堆上面的内存。不回收栈上面空间。很多书上都说,java回收机制是通过判断堆上的对象是否被引用。是否是有栈上的句柄指向堆上的对象,如果没有。java就会自动回收。举一个网上内存溢出非常多的例子来说:

public class out{
  public static void main(String args[]){
       List v=new ArrayList();
       for(int i=0;i<4000000;i++){
          Object o=new Object();
          v.add(o);
         o=null;//强制释放内存
       } 
 }
} 
           

我个人认为。最后令o=null就已经释放了o在堆上占据的内存。我现在的疑问是v里面有4000000个 object,在add进去后。堆内存上肯定还存在4000000个object。我个人觉得最后一句o=null没有什么作用。本人做了个测试。在i=5089378时,内存溢出。如果我屏蔽最后那句o=null 时测试还是i=5089378时内存溢出。这好像证明最后那句强制释放内存没有啥作用。我现在最大的疑问是在list add进去对象后的内存分配情况。到底是怎样。本人非常疑惑。本人对于for循环里面new对象也有很大的疑问。

for(int i=0;i<4000000;i++){
          Object o=new Object();
          v.add(o);
            } 
           

如果你在程序里面写这样的语句:

Object o=new Object();

Object o=new Object();

Object o=new Object();

不用想肯定不会通过编译。但是在for循环里面 new无数个都可以。本人的理解就是for循环里面的循环new对象后,每次新new 的时候上个new出来就会被垃圾自动回收。但是这个引用还存在啊,这么会被自动回收了, 本人今天突然觉悟,发现自己想通了,原来是局部变量,和全局变量的原因。

-- 局部变量 new 出来时,在栈空间和堆空间中分配空间,当局部变量生命周期结束后,栈空间立刻被回收,堆空间区域等待GC回收;

在程序:

public class out{
  public static void main(String args[]){
       List v=new ArrayList();
       for(int i=0;i<4000000;i++){
          Object o=new Object();
          v.add(o);
          o=null;//毫无意义
       } 
 }
} 
           

o是局部变量,new出来后,在作用域内,在栈空间,和堆空间分配。当离开循环,再次进入下一次循环,生命周期结束。栈上的引用空间,立刻被回收。但是在堆空间,new 出来的对象

还被v以用,所以垃圾回收,不会回收,其实本人一直对最后那句o=null很不解。有什么必要吗,垃圾回收立刻回收了栈空间的数据,还有必要吗?画蛇添足。搞的我一直晕头转向。其实这个程序的内存溢出是,v里面放入了过多的object,而导致的溢出。并不是什么垃圾回收的问题。个人认为。呵呵!关于for循环里面new对象的问题,同样的道理,局部变量超过生命周期,自动回收,所以 不管你new多少个相同名称的对象,它在确定的时刻,只有一个引用在栈上。所以就不存在出现两个相同名称的引用。