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多少个相同名称的对象,它在确定的时刻,只有一个引用在栈上。所以就不存在出现两个相同名称的引用。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值