Feign配置RequestContextHolder.getRequestAttributes()为null的解决方案

Feign配置RequestContextHolder.getRequestAttributes报NullPointException

  • 场景
  • 分析原因
    • 关于RequestContextHolder
    • 原因
    • 解决方案

场景

feign配置如下
Feign配置RequestContextHolder.getRequestAttributes()为null的解决方案_第1张图片
异步代码中调用feign
Feign配置RequestContextHolder.getRequestAttributes()为null的解决方案_第2张图片
当用在异步代码中用feign调用微服务时,后台日志中出现报错信息NullPoinntException

Feign配置RequestContextHolder.getRequestAttributes()为null的解决方案_第3张图片

分析原因

关于RequestContextHolder

实现原理 ThreadLocal
ThreadLocal是一个本地线程副本变量工具类。主要用于将私有线程和该线程存放的副本对象做一个映射,各个线程之间的变量互不干扰,在高并发场景下,可以实现无状态的调用,特别适用于各个线程依赖不同的变量值完成操作的场景。

在每个线程的内部存在一个数据结构为Map的ThreadLocals变量,以的形式保存着线程变量和其对应的值。

对于不同的线程,每次获取副本值时,别的线程并不能获取到当前线程的副本值,形成了副本的隔离,互不干扰。

原因

RequestContextHolder将当前线程中的request信息存储在ThreadLocal中,开启另外一个线程异步调用feign,httpRequest是绑定在ThreadLocal中的,只能在当前线程中共享,子线程是不能获取到父线程的属性。

解决方案

1、取消异步调用,改为同步调用
2、沿用异步调用,但是在调用前,把父线程request信息绑定给子线程,在方法执行的前面,加入以下语句
下面展示一些 内联代码片

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

你可能感兴趣的:(避坑指南,java,多线程,rpc)