Feign异步模式丢失上下文问题

Feign异步模式丢失上下文问题

问题描述

当我们使用异步对我们代码进行操作优化时,代码中使用了RequestContextHolder去获取上下文的数据,当我们执行原来可以执行的业务时发现报了空指针异常或数据为空,这是为什么呢?

原理解释

通过源码我们可以看出来,RequestContextHolder本质的是使用了ThreadLocal作为上下文的实现方式,但ThreadLocal只在自己线程中才可以读取到数据,但我们开启了异步线程,导致数据在不同的线程中为空,那我们怎么解决呢?

解决方法

1、在主线程中读取出主线程的数据,通过RequestContextHolder将数据注入到子线程中即可解决此问题

java 复制代码
@Override
    public OrderConfirmVo confirmOrder() throws ExecutionException, InterruptedException {
        OrderConfirmVo confirmVo = new OrderConfirmVo();
        MemberLoginTo memberLoginTo = loginToThreadLocal.get();
        System.out.println("主线程的id:"+Thread.currentThread().getId());
        // 主线程的threadLocal数据,注意threadLocal中的数据只是在本线程中生效,若启用异步线程则会出现线程读取不到数据的问题
        RequestAttributes requestAttributes = RequestContextHolder.getRequestAttributes();
        // 1、远程查询所有的收货地址的列表
        CompletableFuture<Void> addressFuture = CompletableFuture.runAsync(() -> {
            // 将主线程的threadLocal共享到子线程,避免出现读取不到子线程上下文数据的问题
            System.out.println("当前线程的id:"+Thread.currentThread().getId());
            RequestContextHolder.setRequestAttributes(requestAttributes);
            List<MemberAddressVo> address = memberFeignService.getAddress(memberLoginTo.getId());
            confirmVo.setAddress(address);
        }, executor);
        CompletableFuture<Void> CartItemFuture = CompletableFuture.runAsync(() -> {
            // 2、远程查询购物车的购物项列表
            // 将主线程的threadLocal共享到子线程,避免出现读取不到子线程上下文数据的问题
            System.out.println("当前线程的id:"+Thread.currentThread().getId());
            RequestContextHolder.setRequestAttributes(requestAttributes);
            List<OrderItemVo> userCartItems = cartFeignService.getUserCartItems();
            confirmVo.setOrderItems(userCartItems);
            // feign 在远程调用之前要构造请求,会调用很多的拦截器
        }, executor);
        return confirmVo;
    }

2、在主线程中使用RequestContextHolder.setRequestAttributes(requestAttributes,true);将主线程的数据共享即可解决此问题

java 复制代码
@Override
    public OrderConfirmVo confirmOrder() throws ExecutionException, InterruptedException {
        OrderConfirmVo confirmVo = new OrderConfirmVo();
        MemberLoginTo memberLoginTo = loginToThreadLocal.get();
        System.out.println("主线程的id:"+Thread.currentThread().getId());
        // 主线程的threadLocal数据,注意threadLocal中的数据只是在本线程中生效,若启用异步线程则会出现线程读取不到数据的问题
        RequestAttributes requestAttributes = RequestContextHolder.getRequestAttributes();
        // 开启线程数据共享
        RequestContextHolder.setRequestAttributes(requestAttributes,true);
        // 1、远程查询所有的收货地址的列表
        CompletableFuture<Void> addressFuture = CompletableFuture.runAsync(() -> {
            // 将主线程的threadLocal共享到子线程,避免出现读取不到子线程上下文数据的问题
            System.out.println("当前线程的id:"+Thread.currentThread().getId());
//            RequestContextHolder.setRequestAttributes(requestAttributes);
            List<MemberAddressVo> address = memberFeignService.getAddress(memberLoginTo.getId());
            confirmVo.setAddress(address);
        }, executor);
        CompletableFuture<Void> CartItemFuture = CompletableFuture.runAsync(() -> {
            // 2、远程查询购物车的购物项列表
            // 将主线程的threadLocal共享到子线程,避免出现读取不到子线程上下文数据的问题
            System.out.println("当前线程的id:"+Thread.currentThread().getId());
//            RequestContextHolder.setRequestAttributes(requestAttributes);
            List<OrderItemVo> userCartItems = cartFeignService.getUserCartItems();
            confirmVo.setOrderItems(userCartItems);
            // feign 在远程调用之前要构造请求,会调用很多的拦截器
        }, executor);
        return confirmVo;
    }

要构造请求,会调用很多的拦截器

}, executor);

return confirmVo;

}

复制代码
相关推荐
m0_740043731 天前
【无标题】
java·spring boot·spring·spring cloud·微服务
编程彩机1 天前
互联网大厂Java面试:从微服务到分布式缓存的技术场景解析
redis·spring cloud·消息队列·微服务架构·openfeign·java面试·分布式缓存
Anastasiozzzz1 天前
Nginx和Ribbon的区别
后端·spring cloud·ribbon
码农水水2 天前
从 OpenFeign 到 RestClient:Spring Cloud 新时代的轻量化 HTTP 调用方案
java·运维·后端·spring·http·spring cloud·面试
what丶k2 天前
SpringBoot3 配置文件使用全解析:从基础到实战,解锁灵活配置新姿势
java·数据库·spring boot·spring·spring cloud
小信丶2 天前
@Activate 注解详解:应用场景与实战示例
java·spring boot·后端·spring·spring cloud·微服务·dubbo
编程彩机2 天前
互联网大厂Java面试:从Spring MVC到微服务架构场景解析
java·spring cloud·微服务·分布式事务·spring mvc
鸽鸽程序猿2 天前
【JavaEE】【SpringCloud】 熔断和限流 Alibaba Sentinel
spring cloud·java-ee·sentinel
Roye_ack2 天前
【微服务 Day8】SpringCloud实战开发(Elasticsearch02 + DSL查询、聚合)
spring cloud·微服务·架构·dsl·聚合
努力也学不会java3 天前
【Spring Cloud】注册中心-Nacos
java·人工智能·spring boot·后端·spring·spring cloud