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主线程如何等待子线程执行结束(转)
工作中往往会遇到异步去执行某段逻辑, 然后先处理其他事情, 处理完后再把那段逻辑的处理结果进行汇总的产景, 这时候就需要使用线程了. 一个线程启动之后, 是异步的去执行需要执行的内容的, 不会影响主线 ...
随机推荐
- murmurhash2算法 和 DJB Hash算法是目前最流行的hash算法
murmurhash2算法 和 DJB Hash算法是目前最流行的hash算法 1.DJB HASH算法 1 2 3 4 5 6 7 8 9 10 11 /* the famous DJB Hash ...
- QList, QLinkedList, QVector, QStack, QQueue的区别,以前也没见过QCache,而且可以自定义cost
http://doc.qt.io/qt-4.8/containers.html http://doc.qt.io/qt-4.8/qcache.html
- Android零基础入门第24节:自定义View简单使用
原文:Android零基础入门第24节:自定义View简单使用 当我们开发中遇到Android原生的组件无法满足需求时,这时候就应该自定义View来满足这些特殊的组件需求. 一.概述 很多初入Andr ...
- C#获取字符串宽度像素
通过Graphics对象的MeasureString方法可以获取字符串的大小,如下: Graphics graphics = CreateGraphics(); SizeF sizeF = graph ...
- ORACLE 11.2.0.4 安装在 rhel6 上
. 修改host文件,添加本机的host记录 [root@RACDG ~]# vi /etc/hosts 127.0.0.1 localhost localhost.localdomain local ...
- Homebrew 1.0.0 发布,MacOS 上的包管理器,比如安装qt5keychain
神器,没有它不知道怎么用macos https://www.oschina.net/news/77367/homebrew-1-0-0 Mac OS X用户,qt5keychain可以使用homebr ...
- EF 6.0 Code First 迁移MySql数据库
一.准备工作 使用NUGET安装Entity Framework 6,下载MySql Connector/Net 6.9.5 二.创建实体 我们在下面创建了两个类(博客和文章),并 ...
- maven项目或者SpringBoot项目启动时报错在本地仓库中找不到jar包的解决办法
经常遇到项目检出来后是导入开发工具eclipse中pom文件出错问题,项目启动时遇到了一些列的jar包找不到的问题,所以换个开发平台到IDEA以为会好些,结果同样的问题还是会出现的,为了找到具体的解决 ...
- Spring Boot从入门到实战:集成AOPLog来记录接口访问日志
日志是一个Web项目中必不可少的部分,借助它我们可以做许多事情,比如问题排查.访问统计.监控告警等.一般通过引入slf4j的一些实现框架来做日志功能,如log4j,logback,log4j2,其性能 ...
- sentinel 核心概念
编者注:前段时间笔者在团队内部分享了sentinel原理设计与实现,主要讲解了sentinel基础概念和工作原理,工作原理部分大家听了基本都了解了,但是对于sentinel的几个概念及其之间的关系还有 ...