1、Java的垃圾回收机制
(1)定义
因为系统的内存有限,所以需要通过垃圾回收机制(GC)来对内存进行管理,它可以自动识别回收不再使用的内存,使得开发者不必手动管理内存,减轻了程序员的负担。
(2)什么时候需要使用GC
- 内存不足:当堆内存空间不足时,垃圾回收机制会被触发以回收不再使用的内存
- 触发手动GC:在程序中可以调用 System.gc() 方法来触发垃圾回收机制,但该方法只是向垃圾回收器建议执行一次垃圾回收,而不是强制执行。【不建议频繁手动触发】
- 对象引用变为null:当一个对象的引用被设置为 null 时,垃圾回收器会将该对象标记为可回收对象。
- 对象超出作用域:当一个对象超出它所在作用域时,垃圾回收器会将该对象标记为可回收对象。
(3)如何判断一个对象是否存活?
- 引用计数法: 为每一个对象设置一个整数引用计数器属性,当对象被其他对象引用时,则计数器加;若对象引用被删除,则计数器减1。若某个对象的引用计数器变为0.则该对象就可以被垃圾回收器回收。
- 可达性分析法: 将对象集合作为起点,按照从上到下的方式遍历所有可达对象,标记所有被访问的对象为“存活对象”,最终未被标记的对象被视为“垃圾对象”,需要被回收。具体步骤如下:
1) 从GC Roots出发,遍历所有可达对象。
2)标记所有被访问的对象为“存活对象”。
3)清除所有未被标记的对象,即视为“垃圾对象”,需要被回收。 - 活跃线程持有的对象:一个对象如果被活跃线程所持有,则该对象被视为存活对象。
(4)垃圾回收算法
- 标记-清除算法(Mark and Sweep): 是一种基本的垃圾回收算法,其思路是标记所有被引用的对象,清除未被标记的对象。缺点是会产生内存碎片,需要进行垃圾压缩。
- 复制算法(Copying): 将内存分为两个区域,每次只使用其中一个区域,将存活的对象复制到另一个区域,再将原来的区域全部清空。缺点是浪费了一半的空间,适用于对象生命周期短的场景。
- 标记-整理算法(Mark and Compact): 先标记所有被引用的对象,然后将所有存活对象移到一端,清理另一端的未被标记对象。该算法需要进行垃圾压缩,但不会产生内存碎片。
- 分代收集算法(Generational Collection): 根据对象的生命周期将内存分为几代,不同代的对象使用不同的垃圾回收算法和策略。一般将新创建的对象放到年轻代,年轻代使用复制算法,老年代使用标记-整理算法。
2、Java的内存分区
Java的内存分为以下几个区域:
(1)程序计数器(Program Counter Register):用于记录当前线程执行的位置,如果执行的是本地方法,计数器值为空。
(2)Java虚拟机栈(Java Virtual Machine Stacks):每个线程都有一个私有的栈,用于存储方法执行时的局部变量、操作数栈、方法出口等信息。栈帧中包含局部变量表、操作数栈、动态连接、方法出口等信息。
(3)本地方法栈(Native Method Stack):用于存储本地方法的参数和返回值。
(4)Java堆(Java Heap):用于存储对象实例和数组,是Java程序中最大的一块内存。Java堆可以分为年轻代和老年代,年轻代又可以分为Eden区、Survivor区1和Survivor区2。
(5)方法区(Method Area): 用于存储类的信息、常量、静态变量、编译器编译后的代码等数据。
(6)运行时常量池(Runtime Constant Pool):是方法区的一部分,用于存储编译期生成的各种字面量和符号引用。
其中,程序计数器、Java虚拟机栈、本地方法栈的内存分配和回收都由线程独立完成,Java堆和方法区则是所有线程共享的。Java虚拟机通过垃圾回收器对Java堆中的对象进行自动回收,方法区则是永久代的实现区域,永久代在Java 8以后已经被元空间(Metaspace)所取代。