当前位置: 首页 > news >正文

线程池处理异常

在这里插入图片描述

线程池在提交的任务在处理过程中发生了异常,却没有捕获到,导致异常只是输出在控制台,这通常需要把异常记录下来

1、通过观察ThreadGroup的构造方法知道,当调用线程组的构造方法时,会获取当前线程所属的线程组,作为当前正在创建的线程组的parent线程组。

2、通过在main方法中获取线程组信息可知,当前main线程所属的线程组是main线程组,而main线程组的parent线程组是system线程组。

3ThreadGroup的uncaughtException实现如下
public void uncaughtException(Thread t, Throwable e) {
        if (parent != null) {
            parent.uncaughtException(t, e);
        } else {
            Thread.UncaughtExceptionHandler ueh =
                Thread.getDefaultUncaughtExceptionHandler();
            if (ueh != null) {
                ueh.uncaughtException(t, e);
            } else if (!(e instanceof ThreadDeath)) {
            	/* 【异常输出信息的红色是这里输出来的】 */
                System.err.print("Exception in thread \""
                                 + t.getName() + "\" ");
                e.printStackTrace(System.err);
            }
        }
    }

4Thread类有个dispatchUncaughtException方法,当Thread线程运行run方法时,异常抛出来而没有处理时将会把异常抛给虚拟机,虚拟机将会交给该Thread对象的dispatchUncaughtException方法处理,可以从如下代码看到,如果没有给Thread设置具体的UncaughtExceptionHandler,将会给线程组处理,而线程组一直委托给parent处理,所以最后到了system线程组处理,由于它的parent是null,所以就使用System.err输出了异常信息
private void dispatchUncaughtException(Throwable e) {
      getUncaughtExceptionHandler().uncaughtException(this, e);
}
public UncaughtExceptionHandler getUncaughtExceptionHandler() {
      return uncaughtExceptionHandler != null ?uncaughtExceptionHandler : group;
}

5、线程池中默认的DefaultThreadFactory的实现如下
static class DefaultThreadFactory implements ThreadFactory {
    private static final AtomicInteger poolNumber = new AtomicInteger(1);
    private final ThreadGroup group;
    private final AtomicInteger threadNumber = new AtomicInteger(1);
    private final String namePrefix;

    DefaultThreadFactory() {
        SecurityManager s = System.getSecurityManager();
        /* 交给的是当前线程所属的线程组 */
        group = (s != null) ? s.getThreadGroup() :
                              Thread.currentThread().getThreadGroup();
        namePrefix = "pool-" +
                      poolNumber.getAndIncrement() +
                     "-thread-";
    }

    public Thread newThread(Runnable r) {
    	/* 创建的线程 所指定的线程组是在构造方法中设置的 */
        Thread t = new Thread(group, r,
                              namePrefix + threadNumber.getAndIncrement(),
                              0);
        if (t.isDaemon())
            t.setDaemon(false);
        if (t.getPriority() != Thread.NORM_PRIORITY)
            t.setPriority(Thread.NORM_PRIORITY);
        return t;
    }
}

测试

public class ThrTask implements Runnable {
    private int a, b;

    private ThrTask(int a, int b) {
        this.a = a;
        this.b = b;
    }

    @Override
    public void run() {
        double re = a / b;
        System.out.println(re);
    }

    public static void main(String[] args) {
        ThreadPoolExecutor threadPoolExecutor = new ThreadPoolExecutor(0, Integer.MAX_VALUE, 1, TimeUnit.SECONDS, new SynchronousQueue<>(), new ThreadFactory() {
            @Override
            public Thread newThread(Runnable r) {
                Thread t = new Thread(Thread.currentThread().getThreadGroup(), r, "myThread");
                if (t.isDaemon()) t.setDaemon(false);
                t.setUncaughtExceptionHandler(new Thread.UncaughtExceptionHandler() {
                    @Override
                    public void uncaughtException(Thread t, Throwable e) {
                        System.out.println("线程" + t.getName() + "出现异常" + e);
                        // record this self..

                    }
                });
                return t;
            }
        });
        for (int i = 0; i < 5; i++) {
            //将不会抛出异常
            // threadPoolExecutor.submit(new ThrTask(5, i));
            //将会抛出异常
            threadPoolExecutor.execute(new ThrTask(5, i));
        }
    }
}
http://www.dtcms.com/a/15495.html

相关文章:

  • 应对DeepSeek总是服务器繁忙的解决方法
  • 服务器linux操作系统安全加固
  • 深度求索(DeepSeek)的AI革命:NLP、CV与智能应用的技术跃迁
  • 第一章嵌入式系统概论考点07数字音频与数字视频
  • Python数据可视化 - Matplotlib教程
  • 数据库开发常识(10.6)——考量使用临时表及表连接写法(3)
  • 回环地址127.0.0.1跟自身IP有什么区别?
  • 第二十二章 P - R 开头的术语
  • ThreadLocal源码分析
  • 微信小程序组件间通信与传值的全面解析
  • Node.js使用教程
  • LabVIEW显微镜成像偏差校准
  • Flutter 异步编程利器:Future 与 Stream 深度解析
  • 算法-哈希表篇06-赎金信
  • flutter doctor 报错—CocoaPods not installed
  • STM32操作FLASH
  • C#中的Interface、Abstract和Virtual
  • Linux | 进程相关概念(进程、进程状态、进程优先级、环境变量、进程地址空间)
  • AI智能体,AI computer use:浏览器使用(Browser Use)项目实际体验,ollama deepseek r1
  • 网络安全初级实战笔记(一):owasp zap 暴力破解
  • 学二手书---《python 数据科学手册》学习笔记250213 第一章(一)
  • Ollama + DeepSeek + Dify私有化部署自己的AI Agent
  • 达梦 AWR 生成
  • C# 中用于比较两个字符串的方法string.Compare
  • InfiniBand与IP over InfiniBand(IPOIB):实现高性能网络通信的底层机制
  • 3.3.4 VO-O语法- 算子分类(二)
  • 【DDD系列-3】DDD战术设计实践分享
  • 什么是多光谱环形光源
  • 耐张线夹压接图片智能识别
  • 解码DeepSeek家族系列:大语言模型赛道上的黑马传奇