CompletableFuture的默认线程池ForkJoinPool会导致类加载失败

1、相关代码块

使用静态代码块初始化策略模式

java 复制代码
public class PreviewHandlerFactory {
    public static final Map<String, BasePreviewTypeAbstractHandler> PREVIEW_TYPE_MAP = new HashMap<>();

    static {
        initPreviewTypeHandler();
    }

    private static void initPreviewTypeHandler() {
        Class clazz = BasePreviewTypeAbstractHandler.class;
        Set<Class<?>> classes = null;
        try {
            classes = ScanSupport.classInfos(clazz.getPackage().getName());
        } catch (Exception e) {
            log.error("扫描PreviewTypeAbstractHandler失败:", e);
        }
        if (CollectionUtils.isNotEmpty(classes)) {
            for (Class<?> aClass : classes) {
                if (ScanSupport.validate(aClass.getSuperclass(), clazz)) {
                    try {
                        BasePreviewTypeAbstractHandler handler = (BasePreviewTypeAbstractHandler) aClass.newInstance();
                        PREVIEW_TYPE_MAP.put(handler.type(), handler);
                    } catch (Exception e) {
                        log.error("初始化PreviewTypeAbstractHandler失败:", e);
                    }
                }
            }
        }else {
            log.error("未扫描PreviewTypeAbstractHandle相关处理器");
        }

    }

    public static BasePreviewTypeAbstractHandler getPreviewTypeHandler(PreviewTypeEnum type) throws Exception {
        BasePreviewTypeAbstractHandler previewTypeHandler = PREVIEW_TYPE_MAP.get(type.getName());
        if (previewTypeHandler == null) {
            throw new Exception("预览类型异常");
        }
        return previewTypeHandler;
    }
}

使用CompletableFuture的默认线程池调用getPreviewTypeHandler(PreviewTypeEnum type)方法触发静态代码块进行指定类加载。

java 复制代码
 ClassLoader contextClassLoader1 = Thread.currentThread().getContextClassLoader();
        log.info("contextClassLoader1-----" + contextClassLoader1.toString());
        CompletableFuture.runAsync(() -> {
            try {
                ClassLoader contextClassLoader2 = Thread.currentThread().getContextClassLoader();
                log.info("contextClassLoader2-----" + contextClassLoader2.toString());
                PreviewHandlerFactory.getPreviewTypeHandler(PreviewTypeEnum.PREVIEW_TYPE_ORIGINAL);
            } catch (Exception e) {
                throw new RuntimeException(e);
            }
        });

2、现象

在本地使用idea运行程序,一切正常,可以扫描到预期的类。

上测试环境后,出现问题,扫描不到预期的类。

3、原因分析

本地启动

在idea点击运行按钮运行程序,实际上idea是javac编译后用java命令classpath参数把依赖路径全部都加入命令行。

它的所有依赖类加载器都是使用的是TomcatEmbeddedWebappClassLoader,他的父加载器是AppClassLoader(它可以加载classpath路径下的所有类);

并且在使用CompletableFuture 时,如果不指定自定义线程池,则会使用默认的线程池 ForkJoinPool,在默认的线程工厂中设置了创建线程的时候使用Application ClassLoader(应用程序类加载器) ,可以加载classpath下的类

java -jar方式启动

springboot项目打包后在测试环境上的启动方式是java -jar xxx.jar。

相关结构:

问题来了:普通的 Application ClassLoader 能加载 BOOT-INF/classes/MyApp.class 吗?

不能! 因为标准 ClassLoader 只会从 jar 根目录找类,根本不知道 BOOT-INF 是啥。

Spring Boot 怎么解决的?

它自己实现了一套 LaunchedURLClassLoader,继承自 URLClassLoader,专门用来加载 BOOT-INF 里的内容。

关键逻辑在 org.springframework.boot.loader.LaunchedURLClassLoader:

它会把 BOOT-INF/classes 和 BOOT-INF/lib/*.jar 全部加入自己的搜索路径;

当需要加载 com.example.MyService 时,它直接去 BOOT-INF/classes/com/example/MyService.class 找;

但它依然遵守"双亲委派"!

也就是说:Spring Boot 并没有打破双亲委派,而是在 Application ClassLoader 之下,又加了一层自定义加载器,继续遵守双亲委派规则。

所以会出现CompletableFuture 有些类无法加载(因为默认的CommonJoinPool设置了类加载器为AppClassLoader,无法访问 BOOT-INF/ 路径),委派给父类也无法加载。

LaunchedURLClassLoader加载器是唯一能够加载 BOOT-INF/下类的类加载器。

解决方案

任务内显式指定类加载器

java 复制代码
ClassLoader contextClassLoader1 = Thread.currentThread().getContextClassLoader();
        log.info("contextClassLoader1-----" + contextClassLoader1.toString());
        CompletableFuture.runAsync(() -> {
            try {
                Thread.currentThread().setContextClassLoader(contextClassLoader1);
                ClassLoader contextClassLoader2 = Thread.currentThread().getContextClassLoader();
                log.info("contextClassLoader2-----" + contextClassLoader2.toString());
                PreviewHandlerFactory.getPreviewTypeHandler(PreviewTypeEnum.PREVIEW_TYPE_ORIGINAL);
            } catch (Exception e) {
                throw new RuntimeException(e);
            }
        });

使用自定义线程池

java 复制代码
ClassLoader contextClassLoader1 = Thread.currentThread().getContextClassLoader();
        log.info("contextClassLoader1-----" + contextClassLoader1.toString());
        CompletableFuture.runAsync(() -> {
            try {
//                Thread.currentThread().setContextClassLoader(contextClassLoader1);
                ClassLoader contextClassLoader2 = Thread.currentThread().getContextClassLoader();
                log.info("contextClassLoader2-----" + contextClassLoader2.toString());
                PreviewHandlerFactory.getPreviewTypeHandler(PreviewTypeEnum.PREVIEW_TYPE_ORIGINAL);
            } catch (Exception e) {
                throw new RuntimeException(e);
            }
        }, PreviewTaskHandlerExecutors.executor);
相关推荐
小北方城市网4 小时前
Redis 分布式锁高可用实现:从原理到生产级落地
java·前端·javascript·spring boot·redis·分布式·wpf
六义义4 小时前
java基础十二
java·数据结构·算法
毕设源码-钟学长5 小时前
【开题答辩全过程】以 基于SpringBoot的智能书城推荐系统的设计与实现为例,包含答辩的问题和答案
java·spring boot·后端
笨手笨脚の5 小时前
深入理解 Java 虚拟机-03 垃圾收集
java·jvm·垃圾回收·标记清除·标记复制·标记整理
莫问前路漫漫6 小时前
WinMerge v2.16.41 中文绿色版深度解析:文件对比与合并的全能工具
java·开发语言·python·jdk·ai编程
九皇叔叔6 小时前
【03】SpringBoot3 MybatisPlus BaseMapper 源码分析
java·开发语言·mybatis·mybatis plus
挖矿大亨6 小时前
c++中的函数模版
java·c++·算法
a程序小傲7 小时前
得物Java面试被问:RocketMQ的消息轨迹追踪实现
java·linux·spring·面试·职场和发展·rocketmq·java-rocketmq
青春男大7 小时前
Redis和RedisTemplate快速上手
java·数据库·redis·后端·spring·缓存
Ghost Face...7 小时前
i386 CPU页式存储管理深度解析
java·linux·服务器