Java GC

java 内存区域划分

方法区:

1 主要储存被虚拟机加载的类的信息,常量,静态变量,以及及时编辑器编译后的代码等数据

2 * 被线程共享

3 方法区里有一个 运行常量池 , 大部分用于存放静态编译产生的字面量和符号引用,运行时的常量也可能放在这个常量池里。

虚拟机栈:

1 为Java 方法服务,方法在执行的时候会创建一个栈帧,用于存储 局部变量表,操作数栈,动态链接与方法出口灯,

2 栈为线程私有,生命周期与线程相同。

3. 局部变量表里存储的是基本数据类型、returnAddress类型(指向一条字节码指令的地址)和对象引用,这个对象引用有可能是指向对象起始地址的一

  个指针,也有可能是代表对象的句柄或者与对象相关联的位置。局部变量所需的内存空间在编译器间确定

4.操作数栈的作用主要用来存储运算结果以及运算的操作数,它不同于局部变量表通过索引来访问,而是压栈和出栈的方式

5.每个栈帧都包含一个指向运行时常量池中该栈帧所属方法的引用,持有这个引用是为了支持方法调用过程中的动态连接.动态链接就是将常量池中的符

号引用在运行期转化为直接引用。

本地方法栈:跟虚拟机栈差不多,主要是为Native方法服务。

程序计数器:唯一一个不发生GC的地方 改变计数值执行字节码命令,分支,循环,跳转,异常处理

,线程恢复等,我感觉跟引用计数法有些像。

堆:对象的创建,一个回收常在这里发生。

判断GC对象

1 引用计数法

引用计数法就是给对象设置一个计数器 每当有一个地方引用这个对象 就将计数器加一,引用失效时,计数器就减一。

问题:循环引用,造成对象回收不掉。

2 可达性分析

从一个Gc Root 对象向下搜索,如果一个对象到GcRoot没有任何引用链相连,则说明此对象不可用。

GcRoot 对象

栈中的对象(基本)

方法区类静态属性引用的对象

方法区常量池引用的对象。

本地方法栈引用的对象

理解上面的 四个 需要对内存区域进行了解以及熟知。

满足上述条件后不一定会回收,还要经历两次标记的过程。

第一次标记 是否有必要执行 finealize()方法,当对象没有覆盖finealize()方法或者已经被虚拟机调用过,那么就认为没有必要,

如果有必要执行 finealize()方法,那么会将这个对象放在 F-Queue的队列中,虚拟机触发一个线程去执行,优先级底的线程,

此时Gc会对此对象进行第二次标记,在第二次标记之前,如果这个对象有新的引用,该对象被移除‘即将回收‘队列,如果没有,则等待回收。

Java 垃圾回收算法

1 标记-清除

标记回收的对象,然后统一回收。

优点:简单

缺点:

1 效率低,标记跟清除都效率低

2 导致大量的不连续的内存碎片,导致程序分配较大的对象,没有充足的内存提前进行一次GC(内存抖动)

2 复制算法

将内存分为两个相等的两部分,每次使用一块,当要使用完了,将还存活的对象复制到第二块内存板上,

然后清除第一块内存,然后在把存活的对象复制回来。(缺点:会浪费内存)

改进:

内存区域进行了重新划分, 8:1:1

较大的那块 叫 Eden区,剩下叫 Survior区

每次优先使用Eden区,弱Eden区满,讲对象复制到第二块内存,然后清除Eden区

如果存活的对象太多,以至于Survior不够,会将这些对象通过分配担保机制

复制到老年代中(Java 堆)。

3 标记-整理

区分 标记-清除,为了解决标记—清除 产生大量内存块碎片的问题。

当对象存活率较高的时候,也解决了复制算法的效率问题。

前两个算法的合集吧 在清除无用对象的时候讲对象移到另一端,然后清除掉边界以外的对象,这样就不会产

零碎的内存碎片了。

4 分代回收

根据对象的生命周期,将堆分为新生代跟老年代,(永久代),新生代对象生命周期存活的短(感觉这块是标记)

每次都会有大量的对象死去,采用复制算法,老年代对象存活时间长,所以可以考虑 标记-清除,或者标记—整理。

你可能感兴趣的:(Java GC)