多线程并发调用feign,outcome返回NullPointerException

遇到的问题

看过我之前的文章能发现我从小就有一个并发梦,并且也是对线程协程纤程异步以及并发包有一些了解。这次的接口正好需要多次查询,并发调用feign请求提上日程。

List<Future<CurveLineDataVO>> futureList = Lists.newArrayList();
for (CurveDTO curveDTO : curveDTOList) {
	RequestContextHolder.setRequestAttributes(RequestContextHolder.getRequestAttributes(), true);
    Callable<CurveLineDataVO> callable = () -> getIntervalFlowCurve(curveDTO);
    futureList.add(pool.submit(callable));
}
for (Future<CurveLineDataVO> future : futureList) {
    try {
        curveLineDataVOS.add(future.get());
    } catch (InterruptedException e) {
        log.error("future.get()失败", e);
    } catch (ExecutionException e) {
        throw new RuntimeException(e.getCause().getMessage());
    }
}

可是,future.get一直抛出NullPointerException。
调试发现返回的FutTask对象的

state = 3;
outcome = NullPointerException

查找资料得知原因是HttpServletRequest 为null

Spring Boot 默认使用ThreadLocal把Request设置进请求线程中,这样如果在请求方法里面另起一个子线程然后再通过getRequestAttributes方法获取,是获取不到的。
。。。。。。。。。。。。。。。。。。
在Spring cloud微服务中,feign开启了熔断器(hystrix):feign.hystrix.enabled=ture,并且使用默认的信号隔离级别,、HttpServletRequest对象在父线程与子线程是相互独立的,不共享的。所以子线程中使用父线程的HttpServletRequest数据为null。

如上,加一句问题确实得到了解决。

RequestContextHolder.setRequestAttributes(RequestContextHolder.getRequestAttributes(), true);

但是并没有完全解决:用线程池时,新创建的线程能传入token,线程已经存在被再次利用时还是拿不到。这时需要结合threadLocal,Feign的拦截器,来解决问题。

总结

  1. RequestContextHolder.setRequestAttributes(RequestContextHolder.getRequestAttributes(), true);是通过InheritableThreadLocal实现的。
  2. 线程池中使用InheritableThreadLocal会有问题,因为InheritableThreadLocal的原理是在创建线程时,复制父线程的threadlocal到子线程。当线程创建好后,进行复用时,他的本地threadlocal不会被更新。
  3. 在spring web请求生命周期中,请求结束,即主线程执行完毕时,会清空threadlocal中保存的请求上下文RequestContextHolder。于是,也会清空了原来线程池里的子线程继承过来的副本,导致子线程再拿该数据时为null

三次请求:for循环与线程池加callable的效率差距有一倍。
并发调用feign

线程池ThreadPoolExecutor参数设置

解决了调用不通的问题后,就该解决如何设置线程池来提高资源利用性价比。
corePoolSize、maxPoolSize、queueCapacity。
核心线程数、最大线程数、队列容量对线程池是否继续创建线程的影响关系如下:

  1. 当线程数小于核心线程数时,创建线程。
  2. 当线程数大于等于核心线程数,且任务队列未满时,将任务放入任务队列。
  3. 当线程数大于等于核心线程数,且任务队列已满
    1. 若线程数小于最大线程数,创建线程
    2. 若线程数等于最大线程数,抛出异常,拒绝任务
这方面还需要继续学习。以下是这次实践的一些体会。

阿里java开发规范说不要用Executors创建线程池。

Executors.newFixedThreadPool()

由于这个接口调用才会用到线程池,所以将corePoolSize设置为大于0,没任务也会存活,难免觉得浪费资源。
但是当corePoolSize=0时,其他参数怎么设置,效率都上不来,例如:

private static final ExecutorService  pool = new ThreadPoolExecutor(
0, 20, 5L, TimeUnit.SECONDS, 
new LinkedBlockingQueue<>());
原因是

他会将任务加入队列,然后创建一个线程执行,队列不满不会创建更多的线程去执行。导致线程池只有一个线程活跃。
线程池
如图,只有thread1.
于是,我想着,若是队列满了会怎样呢,把队列的容量设置为1

private static final ExecutorService  pool = new ThreadPoolExecutor(
0, 20, 5L, TimeUnit.SECONDS, 
new LinkedBlockingQueue<>(1));

执行线程确实多了,效率提升明显。
执行线程
但是这样的设置也有一个致命的缺陷。
就是当maxPoolSize太小了,任务数量太多,就会被拒绝而报错。

java.util.concurrent.RejectedExecutionException
Task java.util.concurrent.FutureTask@53b8ecad rejected from java.util.concurrent.ThreadPoolExecutor@1ad994af

allowCoreThreadTimeout

还有一个allowCoreThreadTimeout参数,或许能达到我想让线程池不断给我创建新的线程去执行,同时在空闲时也不占用资源。
allowCoreThreadTimeout = true
能让核心线程在超时后退出,直到核心线程数量=0

最后决定

private static final ThreadPoolExecutor pool;

static {
    pool = new ThreadPoolExecutor(3, 5, 5L, TimeUnit.SECONDS, new LinkedBlockingQueue<>());
    pool.allowCoreThreadTimeOut(true);
}
  • 1
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
如果在RabbitListener中调用Feign时出现NullPointerException,可能是由于Feign的实例未能正确注入导致的。 请检查以下几个方面: 1. 确保你在RabbitMQ消费者类(即RabbitListener所在的类)上使用了@Component或@Service等注解,以便将其纳入Spring容器的管理。 2. 确保Feign的相关依赖已经正确添加,并且Feign的配置文件(如application.properties或application.yml)已经正确配置。 3. 确保Feign的接口定义正确,包括方法名、参数列表和返回类型等。可以尝试在RabbitListener方法中打印一些日志,检查Feign实例是否为null,或者尝试调用其他Feign接口方法,看是否能正常返回数据。 4. 如果以上步骤都没有问题,可以尝试使用@Autowired或@Resource等注解将Feign的实例直接注入到RabbitMQ消费者类中。在RabbitListener方法中直接使用注入的Feign实例进行调用。 示例代码如下: ```java import org.springframework.amqp.rabbit.annotation.RabbitListener; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Component; @Component public class RabbitMQConsumer { @Autowired private YourFeignClient feignClient; @RabbitListener(queues = "your-queue") public void handleMessage(String message) { // 调用Feign接口 feignClient.yourMethod(); } } ``` 如果以上方法仍然无法解决问题,请提供更多的错误信息和代码,以便我们更好地帮助你解决问题。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值