java高并发编程--04--Hook线程以及捕获线程执行异常

Posted shouwangyixin

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了java高并发编程--04--Hook线程以及捕获线程执行异常相关的知识,希望对你有一定的参考价值。

1.获取线程运行时异常
Thread类处理运行时异常的四个API:
public void setUncaughtExceptionHandler(UncaughtExceptionHandler eh):为某个线程UncaughtExceptionHandler
public static setDefaultUncaughtExceptionHandler(UncaughtExceptionHandler eh):设置全局UncaughtExceptionHandler
public UncaughtExceptionHandler getUncaughtExceptionHandler():获取特定线程的UncaughtExceptionHandler
public static UncaughtExceptionHandler getDefaultUncaughtExceptionHandler():获取全局UncaughtExcepitonHandler

1.UncaughtExceptionHandler简介
线程在执行单元中不允许抛出checked异常,线程运行在自己的上下文中,派生它的线程无法直接获取到它运行的异常信息,对此,java提供了一个UncaughtExceptionHandler接口,当线程在运行过程中出现异常时会调用UncaughtExceptionHandler接口,从而得知那个线程在运行时出错以及出现什么样的错误。
UncaughtExceptionHandler接口是Thread类的一个内部接口:

@FunctionalInterface
public interface UncaughtExceptionHandler 
    /**
     * Method invoked when the given thread terminates due to the
     * given uncaught exception.
     * <p>Any exception thrown by this method will be ignored by the
     * Java Virtual Machine.
     * @param t the thread
     * @param e the exception
     */
    void uncaughtException(Thread t, Throwable e);

该接口是一个函数式接口,只有一个方法,会被Thread类的dispatchUncaughtException方法调用:

/**
 * Dispatch an uncaught exception to the handler. This method is
 * intended to be called only by the JVM.
 */
private void dispatchUncaughtException(Throwable e) 
    getUncaughtExceptionHandler().uncaughtException(this, e);

当线程出现异常时,会调用dispatchUncaughtException方法,将对应的线程实例及异常信息传递给回调接口
使用代码示例如下:

public class UncaughtExceptionHandlerTest 
    public static void main(String[] args) 
        //设置回调接口实现
        Thread.setDefaultUncaughtExceptionHandler((t,e)->
            System.out.println("--->线程"+t.getName()+"发生异常,异常信息如下:");
            e.printStackTrace();
        );
        new Thread(()->
            try 
                TimeUnit.SECONDS.sleep(2);
             catch (InterruptedException e1) 
                e1.printStackTrace();
            
            //这里将抛出unchecked异常
            System.out.println(2/0);
        ).start();
    

输出结果:

--->线程Thread-0发生异常,异常信息如下:
java.lang.ArithmeticException: / by zero
    at cp7.cp1.UncaughtExceptionHandlerTest.lambda$1(UncaughtExceptionHandlerTest.java:19)
    at java.base/java.lang.Thread.run(Thread.java:834)

2.Hook线程
2.1Hook线程简介
向jvm程序注入一个Hook线程,在jvm程序退出时,Hook线程会启动执行,可以通过Runtime向jvm程序注入多个Hook线程:

public class HookTest 
    public static void main(String[] args) 
        //注入5个Hook线程
        for(int i = 0;i < 5;i ++) 
            int x = i;
            Runtime.getRuntime().addShutdownHook(new Thread(()->
                System.out.println("hook" + x + " run");
                try 
                    TimeUnit.SECONDS.sleep(2);
                 catch (InterruptedException e) 
                    e.printStackTrace();
                
                System.out.println("hook" + x + " is over");
            ));
        
        try 
            TimeUnit.SECONDS.sleep(2);
         catch (InterruptedException e) 
            e.printStackTrace();
        
        System.out.println("main is over");
    


输出结果:
main is over
hook2 run
hook1 run
hook0 run
hook4 run
hook3 run
hook1 is over
hook4 is over
hook0 is over
hook3 is over
hook2 is over
2.2Hook线程使用与注意事项
Hook线程可以阻止程序重复启动,在进程启动时创建一个look文件,创建前先判断这个文件释放存在,如果存在则认为已经启动,程序结束时利用Hook线程的特点删除这个文件。如mysql、zookeeper、kafka等软件都可以看到lock文件的存在
注意:
1)Hook线程只有在收到退出信号时执行,如果使用kill -9 命令结束进程,Hook线程不会执行,lock文件不会被清理
2)Hook线程可以做一些资源释放工作,如关闭文件句柄、socket链接、数据库connection等
3)尽量不要在Hook线程中执行一些耗时非常长的操作,否则程序迟迟无法退出

以上是关于java高并发编程--04--Hook线程以及捕获线程执行异常的主要内容,如果未能解决你的问题,请参考以下文章

Java多线程系列:线程的五大状态,以及线程之间的通信与协作

Java多线程系列:线程的五大状态,以及线程之间的通信与协作

多线程高并发,java并发编程实战pdf下载

Java开发自学教程!japonensisjava东莞

Java高并发学习笔记:Thread详解

Java高级工程师进阶学习:kafka应用场景