8.2 Synchronized、Volatile

volatile

保证变量在多个线程之间可见.

保证可见性的方法:

volatile

synchronized 重量级的锁,

final.

指令重排:a=1; b=2;

有序性

线程内串行语义:

写后读 a=1,b=a;写一个变量后 在读取这个变量

写后写 a=1,a=2;

读后写 a=b,b=1;

怎么保证有序性.

Trace跟踪参数:

-verbose:gc 打印GC日志的

-XX:+PrintGCDetails 打印日志的详细信息

-Xloggc:d:/gc.log 打印日志的路径

-XX:+PrintHeapAtGC 发生GC的时候,是否要重新打印堆的信息

-XX:+TraceClassLoading 类加载的轨迹。 先加载rt.jar,最后加载自己应用的ClassLoader

Heap内存分配参数:

-Xmx 堆的最大值。 -Xmx20m

-Xms 堆可分配的最小值。 -Xms5m

Heap Memory 是如何分配的?

案例

-Xmn 新生代的大小 eden+2s。

-XX:NewRatio 新生代(eden+2s)/老年代。1/4

-XX:SurvivorRatio 幸存区的比例。2s/eden。

案例及GC变化

只要没有一块连续的空间存储1MB,则会发生GC。

GC 表示发生GC的类型 GC|FULLGC。

FULLGC: stop the world,把所有的用户线程停止。

PSYoungGen:年轻代收集器。

11279K->1528K(13824K)

gc前该内存区域(年轻代)已经使用的内存->gc后该内存区域已使用的容量(该内存区域总内存)。

JVM内存回收:

1.标记-清除算法

(1)标记阶段,从根节点不可达的对象做标记。

(2)清除阶段。清除

(3)优点:清除特别快

(3)缺点:导致内存不连续,会产生碎片。

2.复制算法

为了优化标记清除算法产生的碎片。

原理:

优缺点:不会产生内存碎片

GC,FULL GC 就是使用的复制算法

3.标记整理算法

让存活的对象向一端移动。直接清理到这个边界之后的所有对象。不会产生碎片。

缺点:

分代收集算法

分代思想

新生代(空间大)可以用复制算法;老年代(不容易死的对象),因为空间小,所以用标记清除算法或标记整理算法。

JVM垃圾收集器:

..........年轻代.............................

Serial:串行收集器.stop the world,会导致服务(例如线程)停顿。使用的是 新生代是(复制算法)老年代是(标记整理算法)。

串行收集器:一个是用户线程,一个是GC线程,一旦GC垃圾回收,用户线程必须暂停。

ParNew:并行收集器.多个GC线程可以并行执行。也会发生stop the world。

新生代使用并行收集器,老年代使用串行收集器。

Parallel:

Parallel Scavenge:(基本满足需求)并行收集器:专门对于新生代的收集器。多线程的,使用复制算法,更加关注吞吐量,减少stop the world的时间,减少GC的时间。

...........老年代...........................

Serial Old:老年代的串行收集器,

Parallel Old:老年代的并行处理器,

CMS:真正意义上的并发收集器

并发:用户线程和GC线程可以同时执行,如果发生GC,用户线程依然可以执行。

并行:用户线程和GC线程可以同时执行,如果发生GC,用户线程会暂停。

G1:可以设置GC的时间.

分代收集是如何组合的?

新生代:Serial, ParNew, Parallel Scavenge

老年代:Serial Old, Parallel Old, CMS

8.2 Synchronized、Volatile_第1张图片
图片发自App

你可能感兴趣的:(8.2 Synchronized、Volatile)