mirror of
https://github.com/Snailclimb/JavaGuide
synced 2025-06-16 18:10:13 +08:00
修改复制算法的描述为标记-复制算法
This commit is contained in:
parent
9c1f032bf8
commit
dfc7c8d15d
@ -28,7 +28,7 @@
|
|||||||
|
|
||||||
**代理模式的主要作用是扩展目标对象的功能,比如说在目标对象的某个方法执行前后你可以增加一些自定义的操作。**
|
**代理模式的主要作用是扩展目标对象的功能,比如说在目标对象的某个方法执行前后你可以增加一些自定义的操作。**
|
||||||
|
|
||||||
举个例子:你找了小红来帮你文化,小红就可以看作是代理你的代理对象,代理的行为(方法)是问话。
|
举个例子:你找了小红来帮你问话,小红就可以看作是代理你的代理对象,代理的行为(方法)是问话。
|
||||||
|
|
||||||

|

|
||||||
|
|
||||||
|
@ -1,5 +1,3 @@
|
|||||||
|
|
||||||
|
|
||||||
<!-- @import "[TOC]" {cmd="toc" depthFrom=1 depthTo=6 orderedList=false} -->
|
<!-- @import "[TOC]" {cmd="toc" depthFrom=1 depthTo=6 orderedList=false} -->
|
||||||
|
|
||||||
<!-- code_chunk_output -->
|
<!-- code_chunk_output -->
|
||||||
@ -23,7 +21,7 @@
|
|||||||
- [2.6 如何判断一个类是无用的类](#26-如何判断一个类是无用的类)
|
- [2.6 如何判断一个类是无用的类](#26-如何判断一个类是无用的类)
|
||||||
- [3 垃圾收集算法](#3-垃圾收集算法)
|
- [3 垃圾收集算法](#3-垃圾收集算法)
|
||||||
- [3.1 标记-清除算法](#31-标记-清除算法)
|
- [3.1 标记-清除算法](#31-标记-清除算法)
|
||||||
- [3.2 复制算法](#32-复制算法)
|
- [3.2 标记-复制算法](#32-标记-复制算法)
|
||||||
- [3.3 标记-整理算法](#33-标记-整理算法)
|
- [3.3 标记-整理算法](#33-标记-整理算法)
|
||||||
- [3.4 分代收集算法](#34-分代收集算法)
|
- [3.4 分代收集算法](#34-分代收集算法)
|
||||||
- [4 垃圾收集器](#4-垃圾收集器)
|
- [4 垃圾收集器](#4-垃圾收集器)
|
||||||
@ -89,8 +87,11 @@ Java 堆是垃圾收集器管理的主要区域,因此也被称作**GC 堆(G
|
|||||||
> size_t total = 0;
|
> size_t total = 0;
|
||||||
> uint age = 1;
|
> uint age = 1;
|
||||||
> while (age < table_size) {
|
> while (age < table_size) {
|
||||||
> total += sizes[age];//sizes数组是每个年龄段对象大小
|
> //sizes数组是每个年龄段对象大小
|
||||||
> if (total > desired_survivor_size) break;
|
> total += sizes[age];
|
||||||
|
> if (total > desired_survivor_size) {
|
||||||
|
> break;
|
||||||
|
> }
|
||||||
> age++;
|
> age++;
|
||||||
> }
|
> }
|
||||||
> uint result = age < MaxTenuringThreshold ? age : MaxTenuringThreshold;
|
> uint result = age < MaxTenuringThreshold ? age : MaxTenuringThreshold;
|
||||||
@ -186,8 +187,11 @@ public class GCTest {
|
|||||||
> size_t total = 0;
|
> size_t total = 0;
|
||||||
> uint age = 1;
|
> uint age = 1;
|
||||||
> while (age < table_size) {
|
> while (age < table_size) {
|
||||||
> total += sizes[age];//sizes数组是每个年龄段对象大小
|
> //sizes数组是每个年龄段对象大小
|
||||||
> if (total > desired_survivor_size) break;
|
> total += sizes[age];
|
||||||
|
> if (total > desired_survivor_size) {
|
||||||
|
> break;
|
||||||
|
> }
|
||||||
> age++;
|
> age++;
|
||||||
> }
|
> }
|
||||||
> uint result = age < MaxTenuringThreshold ? age : MaxTenuringThreshold;
|
> uint result = age < MaxTenuringThreshold ? age : MaxTenuringThreshold;
|
||||||
@ -225,7 +229,7 @@ public class GCTest {
|
|||||||
|
|
||||||
## 2 对象已经死亡?
|
## 2 对象已经死亡?
|
||||||
|
|
||||||
堆中几乎放着所有的对象实例,对堆垃圾回收前的第一步就是要判断那些对象已经死亡(即不能再被任何途径使用的对象)。
|
堆中几乎放着所有的对象实例,对堆垃圾回收前的第一步就是要判断哪些对象已经死亡(即不能再被任何途径使用的对象)。
|
||||||
|
|
||||||

|

|
||||||
|
|
||||||
@ -262,6 +266,7 @@ public class ReferenceCountingGc {
|
|||||||
- 本地方法栈(Native 方法)中引用的对象
|
- 本地方法栈(Native 方法)中引用的对象
|
||||||
- 方法区中类静态属性引用的对象
|
- 方法区中类静态属性引用的对象
|
||||||
- 方法区中常量引用的对象
|
- 方法区中常量引用的对象
|
||||||
|
- 所有被同步锁持有的对象
|
||||||
|
|
||||||
### 2.3 再谈引用
|
### 2.3 再谈引用
|
||||||
|
|
||||||
@ -342,9 +347,9 @@ JDK1.2 以后,Java 对引用的概念进行了扩充,将引用分为强引
|
|||||||
|
|
||||||

|

|
||||||
|
|
||||||
### 3.2 复制算法
|
### 3.2 标记-复制算法
|
||||||
|
|
||||||
为了解决效率问题,“复制”收集算法出现了。它可以将内存分为大小相同的两块,每次使用其中的一块。当这一块的内存使用完后,就将还存活的对象复制到另一块去,然后再把使用的空间一次清理掉。这样就使每次的内存回收都是对内存区间的一半进行回收。
|
为了解决效率问题,“标记-复制”收集算法出现了。它可以将内存分为大小相同的两块,每次使用其中的一块。当这一块的内存使用完后,就将还存活的对象复制到另一块去,然后再把使用的空间一次清理掉。这样就使每次的内存回收都是对内存区间的一半进行回收。
|
||||||
|
|
||||||

|

|
||||||
|
|
||||||
@ -358,7 +363,7 @@ JDK1.2 以后,Java 对引用的概念进行了扩充,将引用分为强引
|
|||||||
|
|
||||||
当前虚拟机的垃圾收集都采用分代收集算法,这种算法没有什么新的思想,只是根据对象存活周期的不同将内存分为几块。一般将 java 堆分为新生代和老年代,这样我们就可以根据各个年代的特点选择合适的垃圾收集算法。
|
当前虚拟机的垃圾收集都采用分代收集算法,这种算法没有什么新的思想,只是根据对象存活周期的不同将内存分为几块。一般将 java 堆分为新生代和老年代,这样我们就可以根据各个年代的特点选择合适的垃圾收集算法。
|
||||||
|
|
||||||
**比如在新生代中,每次收集都会有大量对象死去,所以可以选择复制算法,只需要付出少量对象的复制成本就可以完成每次垃圾收集。而老年代的对象存活几率是比较高的,而且没有额外的空间对它进行分配担保,所以我们必须选择“标记-清除”或“标记-整理”算法进行垃圾收集。**
|
**比如在新生代中,每次收集都会有大量对象死去,所以可以选择”标记-复制“算法,只需要付出少量对象的复制成本就可以完成每次垃圾收集。而老年代的对象存活几率是比较高的,而且没有额外的空间对它进行分配担保,所以我们必须选择“标记-清除”或“标记-整理”算法进行垃圾收集。**
|
||||||
|
|
||||||
**延伸面试问题:** HotSpot 为什么要分为新生代和老年代?
|
**延伸面试问题:** HotSpot 为什么要分为新生代和老年代?
|
||||||
|
|
||||||
@ -376,7 +381,7 @@ JDK1.2 以后,Java 对引用的概念进行了扩充,将引用分为强引
|
|||||||
|
|
||||||
Serial(串行)收集器是最基本、历史最悠久的垃圾收集器了。大家看名字就知道这个收集器是一个单线程收集器了。它的 **“单线程”** 的意义不仅仅意味着它只会使用一条垃圾收集线程去完成垃圾收集工作,更重要的是它在进行垃圾收集工作的时候必须暂停其他所有的工作线程( **"Stop The World"** ),直到它收集结束。
|
Serial(串行)收集器是最基本、历史最悠久的垃圾收集器了。大家看名字就知道这个收集器是一个单线程收集器了。它的 **“单线程”** 的意义不仅仅意味着它只会使用一条垃圾收集线程去完成垃圾收集工作,更重要的是它在进行垃圾收集工作的时候必须暂停其他所有的工作线程( **"Stop The World"** ),直到它收集结束。
|
||||||
|
|
||||||
**新生代采用复制算法,老年代采用标记-整理算法。**
|
**新生代采用标记-复制算法,老年代采用标记-整理算法。**
|
||||||
|
|
||||||

|

|
||||||
|
|
||||||
@ -388,7 +393,7 @@ Serial(串行)收集器是最基本、历史最悠久的垃圾收集器了
|
|||||||
|
|
||||||
**ParNew 收集器其实就是 Serial 收集器的多线程版本,除了使用多线程进行垃圾收集外,其余行为(控制参数、收集算法、回收策略等等)和 Serial 收集器完全一样。**
|
**ParNew 收集器其实就是 Serial 收集器的多线程版本,除了使用多线程进行垃圾收集外,其余行为(控制参数、收集算法、回收策略等等)和 Serial 收集器完全一样。**
|
||||||
|
|
||||||
**新生代采用复制算法,老年代采用标记-整理算法。**
|
**新生代采用标记-复制算法,老年代采用标记-整理算法。**
|
||||||
|
|
||||||

|

|
||||||
|
|
||||||
@ -402,7 +407,7 @@ Serial(串行)收集器是最基本、历史最悠久的垃圾收集器了
|
|||||||
|
|
||||||
### 4.3 Parallel Scavenge 收集器
|
### 4.3 Parallel Scavenge 收集器
|
||||||
|
|
||||||
Parallel Scavenge 收集器也是使用复制算法的多线程收集器,它看上去几乎和 ParNew 都一样。 **那么它有什么特别之处呢?**
|
Parallel Scavenge 收集器也是使用标记-复制算法的多线程收集器,它看上去几乎和 ParNew 都一样。 **那么它有什么特别之处呢?**
|
||||||
|
|
||||||
```
|
```
|
||||||
-XX:+UseParallelGC
|
-XX:+UseParallelGC
|
||||||
@ -417,7 +422,7 @@ Parallel Scavenge 收集器也是使用复制算法的多线程收集器,它
|
|||||||
|
|
||||||
**Parallel Scavenge 收集器关注点是吞吐量(高效率的利用 CPU)。CMS 等垃圾收集器的关注点更多的是用户线程的停顿时间(提高用户体验)。所谓吞吐量就是 CPU 中用于运行用户代码的时间与 CPU 总消耗时间的比值。** Parallel Scavenge 收集器提供了很多参数供用户找到最合适的停顿时间或最大吞吐量,如果对于收集器运作不太了解,手工优化存在困难的时候,使用 Parallel Scavenge 收集器配合自适应调节策略,把内存管理优化交给虚拟机去完成也是一个不错的选择。
|
**Parallel Scavenge 收集器关注点是吞吐量(高效率的利用 CPU)。CMS 等垃圾收集器的关注点更多的是用户线程的停顿时间(提高用户体验)。所谓吞吐量就是 CPU 中用于运行用户代码的时间与 CPU 总消耗时间的比值。** Parallel Scavenge 收集器提供了很多参数供用户找到最合适的停顿时间或最大吞吐量,如果对于收集器运作不太了解,手工优化存在困难的时候,使用 Parallel Scavenge 收集器配合自适应调节策略,把内存管理优化交给虚拟机去完成也是一个不错的选择。
|
||||||
|
|
||||||
**新生代采用复制算法,老年代采用标记-整理算法。**
|
**新生代采用标记-复制算法,老年代采用标记-整理算法。**
|
||||||
|
|
||||||

|

|
||||||
|
|
||||||
@ -471,7 +476,7 @@ JDK1.8 默认使用的是 Parallel Scavenge + Parallel Old,如果指定了-XX:
|
|||||||
|
|
||||||
- **并行与并发**:G1 能充分利用 CPU、多核环境下的硬件优势,使用多个 CPU(CPU 或者 CPU 核心)来缩短 Stop-The-World 停顿时间。部分其他收集器原本需要停顿 Java 线程执行的 GC 动作,G1 收集器仍然可以通过并发的方式让 java 程序继续执行。
|
- **并行与并发**:G1 能充分利用 CPU、多核环境下的硬件优势,使用多个 CPU(CPU 或者 CPU 核心)来缩短 Stop-The-World 停顿时间。部分其他收集器原本需要停顿 Java 线程执行的 GC 动作,G1 收集器仍然可以通过并发的方式让 java 程序继续执行。
|
||||||
- **分代收集**:虽然 G1 可以不需要其他收集器配合就能独立管理整个 GC 堆,但是还是保留了分代的概念。
|
- **分代收集**:虽然 G1 可以不需要其他收集器配合就能独立管理整个 GC 堆,但是还是保留了分代的概念。
|
||||||
- **空间整合**:与 CMS 的“标记--清理”算法不同,G1 从整体来看是基于“标记整理”算法实现的收集器;从局部上来看是基于“复制”算法实现的。
|
- **空间整合**:与 CMS 的“标记-清理”算法不同,G1 从整体来看是基于“标记-整理”算法实现的收集器;从局部上来看是基于“标记-复制”算法实现的。
|
||||||
- **可预测的停顿**:这是 G1 相对于 CMS 的另一个大优势,降低停顿时间是 G1 和 CMS 共同的关注点,但 G1 除了追求低停顿外,还能建立可预测的停顿时间模型,能让使用者明确指定在一个长度为 M 毫秒的时间片段内。
|
- **可预测的停顿**:这是 G1 相对于 CMS 的另一个大优势,降低停顿时间是 G1 和 CMS 共同的关注点,但 G1 除了追求低停顿外,还能建立可预测的停顿时间模型,能让使用者明确指定在一个长度为 M 毫秒的时间片段内。
|
||||||
|
|
||||||
G1 收集器的运作大致分为以下几个步骤:
|
G1 收集器的运作大致分为以下几个步骤:
|
||||||
|
Loading…
x
Reference in New Issue
Block a user