Java的Hook线程及捕获线程执行异常
import java.io.IOException; import java.nio.file.Files; import java.nio.file.Path; import java.nio.file.Paths; import java.nio.file.attribute.PosixFilePermission; import java.nio.file.attribute.PosixFilePermissions; import java.util.Set; import java.util.concurrent.TimeUnit; public class Test { public static void main(String[] args){ // CaptureThreadException.test(); // EmptyException.test(); // ThreadHook.test(); PreventDuplicated.test(); } } /* 7.1 获取线程运行时异常 public void setUncaughtExceptionHandler(UncaughtExceptionHandler eh) :为某个线程指定UncaughtExceptionHandler public UncaughtExceptionHandler getUncaughtExceptionHandler(); :获取某个线程的UncaughtExceptionHandler public static void setDefaultUncaughtExceptionHandler(UncaughtExceptionHandler eh) :设置全局的UncaughtExceptionHandler public static void setDefaultUncaughtExceptionHandler(UncaughtException eh) :获取全局的UncaughtExceptionHandler 函数接口,该回调接口会被Thread中的dispatchUncaughtExcept方法调用,当线程在运行 的过程中出现了异常时,JVM会调用dispatchUncaughtExcept方法,该方法将对应的线程实 例以及异常信息传递给回调接口: public interface UncaughtExceptionHandler{ void uncaughtException(Thread t, Throwable e); } private void dispatchUncaughtExcept(Throwable e){ getUncaughtExceptionHandler().uncaughtException(this.e); } 闲谈: 在平时的工作中,这种设计方式是比较常见的,尤其是那种异步执行方法,不如Google的 guava toolkit就提供了EventBus,在EventBus中事件源和实践的subscriber两者 借助于EventBus实现了完全的解耦合,但是在subscriber执行任务时有可能会出现异常 情况,EventBus同样也是借助一个ExceptionHandler进行回调处理的。 */ class CaptureThreadException{ public static void test() { Thread.setDefaultUncaughtExceptionHandler((t,e)->{ System.out.println(t.getName()+" occur exception"); e.printStackTrace(); }); final Thread t = new Thread(()->{ try{ TimeUnit.SECONDS.sleep(2); } catch (InterruptedException e) { } System.out.println(1/0); },"Test-Thread"); t.start(); } } /* 7.1.3 UncaughtExceptionHandler源码分析 未注入UncaughtExceptionHandler回调接口的情况下,就从ThreadGroup中获取: 1.该ThreadGroup如果有父ThreadGroup,则直接使用父Group的UncaughtException 2.如果设置了全局默认的UncaughtExceptionHandler,则调用UncaughtException 3.既没有父,又没有全局的,则直接将异常的堆栈信息定向到System.err中 案例分析: 由于没有指定,所以寻找过程为: 线程出现异常=>Main Group=>System Group=>System.err */ class EmptyException{ public static void test() { ThreadGroup main = Thread.currentThread().getThreadGroup(); System.out.println(main.getName()); System.out.println(main.getParent()); System.out.println(main.getParent().getParent()); final Thread thread = new Thread(()->{ try{ TimeUnit.SECONDS.sleep(2); } catch (InterruptedException e) { e.printStackTrace(); } System.out.println(1/0); },"Test-Thread"); thread.start(); } } /* 7.2.1 Hook线程介绍 Jvm进程的退出是由于JVM进程没有活跃的非守护线程,或者系统中断了信号。向JVM程序 注入多个Hook线程,在JVM进程退出的时候,Hook线程会启动执行,通过Runtime可以为 JVM注入多个Hook线程。 */ class ThreadHook{ public static void test() { Runtime.getRuntime().addShutdownHook(new Thread(){ public void run(){ try{ System.out.println("The hook thread 1 is running..."); TimeUnit.SECONDS.sleep(1); } catch (InterruptedException e) { e.printStackTrace(); } System.out.println("The hook thread 1 will exit."); } }); Runtime.getRuntime().addShutdownHook(new Thread(){ public void run(){ try{ System.out.println("The hook thread 2 is running..."); TimeUnit.SECONDS.sleep(1); } catch (InterruptedException e) { e.printStackTrace(); } System.out.println("The hook thread 2 will exit."); } }); } } /* 7.2.2 Hook线程实战 需求: 在我们开发中经常会遇到Hook线程,比如为了防止某个程序被重复启动,在进行启动时 会创建一个lock文件,进程收到中断信号的时候会删除这个lock文件,我们在mysql服 务器、zookeeper、kafka等系统中都能看到lock文件的存在,本节,将利用hook线程 的特点,模拟一个防止重复启动的线程。 */ class PreventDuplicated { private final static String LOCK_PATH = "C:\\Users\\Administrator\\Desktop\\JavaAPI"; private final static String LOCK_FILE = ".lock"; private final static String PERMISSIONS = "rw-------"; private static Path getLockFile(){ return Paths.get(LOCK_PATH,LOCK_FILE); } private static void checkRunning(){ Path path = getLockFile(); if(path.toFile().exists()){ throw new RuntimeException("The program already running..."); } /* 很尴尬一点,这个地方貌似是针对Linux的写法,我这个地方出现了问题, 暂时先将这个问题放一下吧。。。 */ Set<PosixFilePermission> perms = PosixFilePermissions.fromString(PERMISSIONS); try { Files.createFile(path,PosixFilePermissions.asFileAttribute(perms)); } catch (IOException e) { e.printStackTrace(); } } public static void test(){ Runtime.getRuntime().addShutdownHook(new Thread(){ @Override public void run() { System.out.println("The program received kill SIGNAL."); getLockFile().toFile().delete(); } }); checkRunning(); while (true) { try{ TimeUnit.MILLISECONDS.sleep(1); System.out.println("program is running..."); } catch (InterruptedException e) { e.printStackTrace(); } } } } /* 7.2.3 Hook线程应用场景及注意事项 1.Hook线程只有在收到退出信号的时候会被执行,如果kill使用了-9,那么 Hook线程不会执行,因此lock文件也不会被清理 2.Hook线程中也可以执行一些资源的释放工作,如关闭文件句柄、socket链接 、数据库connection等 */
《Java高并发编程详解》笔记
Java的Hook线程及捕获线程执行异常的更多相关文章
- C#通过接口与线程通信(捕获线程状态)介绍
C#通过接口与线程通信(捕获线程状态)介绍 摘要:本文介绍C#通过接口与线程通信(捕获线程状态),并提供简单的示例代码供参考. 提示:本文所提到的线程状态变化,并不是指线程启动.暂停.停止,而是说线程 ...
- java并发编程(四) 线程池 & 任务执行、终止源码分析
参考文档 线程池任务执行全过程:https://blog.csdn.net/wojiaolinaaa/article/details/51345789 线程池中断:https://www.cnblog ...
- Java并发-UncaughtExceptionHandler捕获线程异常信息并重新启动线程
Java并发-UncaughtExceptionHandler捕获线程异常信息并重新启动线程 一.捕获异常并重新启用线程 public class Testun { public static voi ...
- 死磕 java线程系列之线程池深入解析——普通任务执行流程
(手机横屏看源码更方便) 注:java源码分析部分如无特殊说明均基于 java8 版本. 注:线程池源码部分如无特殊说明均指ThreadPoolExecutor类. 简介 前面我们一起学习了Java中 ...
- java主线程等待所有子线程执行完毕在执行(常见面试题)
java主线程等待所有子线程执行完毕在执行(常见面试题) java主线程等待所有子线程执行完毕在执行,这个需求其实我们在工作中经常会用到,比如用户下单一个产品,后台会做一系列的处理,为了提高效率,每个 ...
- Java中主线程如何捕获子线程抛出的异常
首先明确线程代码的边界.其实很简单,Runnable接口的run方法所界定的边界就可以看作是线程代码的边界.Runnable接口中run方法原型如下: public void run(); 而所有的具 ...
- 死磕 java线程系列之线程池深入解析——未来任务执行流程
(手机横屏看源码更方便) 注:java源码分析部分如无特殊说明均基于 java8 版本. 注:线程池源码部分如无特殊说明均指ThreadPoolExecutor类. 简介 前面我们一起学习了线程池中普 ...
- 【Java多线程系列四】控制线程执行顺序
假设有线程1/线程2/线程3,线程3必须在线程1/线程2执行完成之后开始执行,有两种方式可实现 Thread类的join方法:使宿主线程阻塞指定时间或者直到寄生线程执行完毕 CountDownLatc ...
- Java主线程如何等待子线程执行结束(转)
工作中往往会遇到异步去执行某段逻辑, 然后先处理其他事情, 处理完后再把那段逻辑的处理结果进行汇总的产景, 这时候就需要使用线程了. 一个线程启动之后, 是异步的去执行需要执行的内容的, 不会影响主线 ...
随机推荐
- Win10《芒果TV》商店版双十一独家大礼,每日前100名用户免费领取7天VIP
为答谢大家对Win10<芒果TV>商店版一年以来一如既往的支持,2016年11月1日-11月30日期间,每天登录<芒果TV>UWP版(最新版本v3.1.3)的前100位用户可领 ...
- 微信小程序把玩(三十四)Audio API
原文:微信小程序把玩(三十四)Audio API 没啥可值得太注意的地方 重要属性: 1. wx.getBackgroundAudioPlayerState(object) 获取播放状态 2.wx.p ...
- windows界面程序设计,设置一个窗口始终在屏幕最前,SetWindowPos函数
有时这种需求还是很必须的,比如现在做的一个登录验证系统,如果在windows登录界面点击到窗口外面,那窗口就会永远隐藏掉没法再启用了.这种情况必须设置该窗口一直在最前. 使用函数SetWindowPo ...
- Android零基础入门第81节:Activity数据传递
在Android开发中,经常要在Activity之间传递数据.前面也学习了Activity和Intent相关基础,接下来一起来学习Activity的数据传递. 一.简介 通过前面的学习知道,Inten ...
- 利用开源软件 Hugin 实现照片的景深合成,使用开源软件 enfuse 做照片的曝光合成
http://blog.csdn.net/liyuanbhu/article/details/53573847 http://blog.csdn.net/liyuanbhu/article/detai ...
- 在Delphi中关于UDP协议的实现
原文地址:在Delphi中关于UDP协议的实现作者:菜心 首先我把UDP无连接协议的套接字调用时序图表示出来 在我把在Delphi中使用UDP协议实现数据通讯收发的实现方法总结如下: 例子描述:下 ...
- Qt给应用程序添加版本信息(对rc文件的设置,可利用QT内置变量)
作者:daodaoliang 时间:2016年7月11日16:12:09 版本:V 0.0.4 邮箱:daodaoliang@yeah.net 0. 环境说明 系统环境: win10 64位 Qt环境 ...
- Qt5图形视图框架的“俄罗斯方块”(使用了QGraphicsView)
Qt5 图形视图框架QGraphicsView 1.图形视图框架包含三大类:场景类(QGraphicsScene),视图类(QGraphicsView),图元类(QGraphicsItem): 2.对 ...
- 【搜索引擎】Solr Suggester 实现全文检索功能-分词和和自动提示
功能需求 全文检索搜索引擎都会有这样一个功能:输入一个字符便自动提示出可选的短语: 要实现这种功能,可以利用solr的SuggestComponent,SuggestComponent这种方法利用Lu ...
- shell多线程之进程间通信
# 这是一个简单的并发程序,有如下要求: # .有两个程序a和b,希望他们能并发执行,以节约时间 # .a和b都是按照日期顺序执行,但b每日程序的前提条件是当日a的程序已经执行完毕 #解决方案: # ...