静态成员的垃圾回收
Posted zbuger
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了静态成员的垃圾回收相关的知识,希望对你有一定的参考价值。
今天在网上看到一个讨论命题:当一个单例的对象长久不用时,会不会被jvm的垃圾收集机制回收。原文链接
下面就对Java中的垃圾回收和静态类型做一些总结:
一、Java中的内存分配
1、stack(栈),用于装变量和引用类型。如基本类型和引用类型的引用变量。
2、heap(堆) ,用于装new出来的值。
3、用来装静态变量的区域。如static变量,字符串常量。
4、用来装代码的区域。
二、垃圾回收时机
1、栈上的变量一旦声明,出作用域即会被回收。
2、堆里的对象,没有任何变量(栈上变量或静态区域的变量)指向这个对象的时候就会被回收,这个对象被标记为“垃圾对象”等待回收GC
3、GC是只回收堆空间,检查定时回收(频率由GLR决定)。
4、调用GC.Collect();“直接”回收对象(必须等GC处理完目前的任务,才来处理该任务)。
三、静态成员的垃圾回收
静态成员一般也可以分为静态基本类型和静态引用类型。
静态基本类型存储在在静态变量区域;静态引用类型的引用存储在静态变量区域,而实例(具体内容)存储在堆上。静态成员只存在一份,静态成员加载时机:类加载的时候(第一次访问),这个类中所有静态成员就会被加载在静态存储区,同时存储在静态变量区域的成员一旦创建,直到程序退出才会被回收。(注:如果是引用类型,如static student myst=new student(),myst=null这时候,在静态存储区里面存的是一个地址(myst),这个地址指向在堆里面创建的student实例对象,当myst=null的时候,在静态存储区里面的变量会一直存在,但是在堆里面的student实例对象因为没有变量指向它,所以会被回收)。因此如果不用的静态引用类型可以通过设置=null方式让GC可以回收其堆上的空间。
四、单例模式中静态成员不会被垃圾回收机制回收
单例模式中静态成员的声明由于静态的机制不会被GC回收,而对应的堆上实例对象在外部无法直接设为null,所以不会被垃圾回收机制回收。当然除非人为地断开单例中静态引用到单例对象的联接,否则jvm垃圾收集器是不会回收单例对象的。
以上是关于静态成员的垃圾回收的主要内容,如果未能解决你的问题,请参考以下文章
java中静态成员变量、实例变量、局部变量何时创建、何时销毁?
LevelDB 源码剖析Compaction模块:Minor CompactionMajor Compaction文件选取执行流程垃圾回收