safepoint是什么

java程序外面有很多很多的java线程,每个java线程又有本人的stack,并且共享了heap。这些线程始终运行呀运行,一直对stack和heap进行操作。

这个时候如果JVM须要对stack和heap做一些操作该怎么办呢?

比方JVM要进行GC操作,或者要做heap dump等等,这时候如果线程都在对stack或者heap进行批改,那么将不是一个稳固的状态。GC间接在这种状况下操作stack或者heap,会导致线程的异样。

怎么解决呢?

这个时候safepoint就出场了。

safepoint就是一个平安点,所有的线程执行到平安点的时候就会去查看是否须要执行safepoint操作,如果须要执行,那么所有的线程都将会期待,直到所有的线程进入safepoint。

而后JVM执行相应的操作之后,所有的线程再复原执行。

safepoint的例子

咱们举个例子,个别safepoint比方容易呈现在循环遍历的状况,还是应用咱们之前做null测试用的例子:

public class TestNull {    public static void main(String[] args) throws InterruptedException {        List<String> list= new ArrayList();    list.add("www.flydean.com");        for (int i = 0; i < 10000; i++)        {            testMethod(list);        }        Thread.sleep(1000);    }    private static void testMethod(List<String> list)    {        list.get(0);    }}

运行后果如下:

标红的就是传说中的safepoint。

线程什么时候会进入safepoint

那么线程什么时候会进入safepoint呢?

一般来说,如果线程在竞争锁被阻塞,IO被阻塞,或者在期待取得监视器锁状态时,线程就处于safepoint状态。

如果线程再执行JNI代码的哪一个时刻,java线程也处于safepoint状态。因为java线程在执行本地代码之前,须要保留堆栈的状态,让后再移交给native办法。

如果java的字节码正在执行,那么咱们不能判断该线程是不是在safepint上。

safepoint是怎么工作的

如果你应用的是hotspot JVM,那么这个safepoint是一个全局的safepoint,也就是说执行Safepoint须要暂停所有的线程。

如果你应用的是Zing,那么能够在线程级别应用safepoint。

咱们能够看到生成的汇编语言中safepoint其实是一个test命令。

test指向的是一个非凡的内存页面地址,当JVM须要所有的线程都执行到safepint的时候,就会对该页面做一个标记。从而告诉所有的线程。

咱们再用一张图来具体阐明:

thread1在收到设置safepoint之前是始终执行的,在收到信号之后还会执行一段时间,而后达到Safepint暂停执行。

thread2先执行了一段时间,而后因为CPU被争夺,闲暇了一段时间,在这段时间外面,thread2收到了设置safepoint的信号,而后thread2取得执行势力,接着继续执行,最初达到safepoint。

thread3是一个native办法,将会始终执行,晓得safepoint完结。

thread4也是一个native办法,它和thread3的区别就在于,thread4在safepoint开始和完结之间完结了,须要将控制器转交给一般的java线程,因为这个时候JVM在执行Safepoint的操作,所以任然须要暂停执行。

在HotSpot VM中,你能够在汇编语言中看到safepoint的两种模式:'{poll}' 或者 '{poll return}' 。

总结

本文具体的解说了JVM中Safepoint的作用,心愿大家可能喜爱。

本文作者:flydean程序那些事

本文链接:http://www.flydean.com/jvm-safepoint2/

本文起源:flydean的博客

欢送关注我的公众号:程序那些事,更多精彩等着您!