本文内容由阿里云实名注册用户自发贡献,版权归原作者所有,阿里云开发者社区不拥有其著作权,亦不承担相应法律责任。具体规则请查看《
最近有读者问了这样一个问题:陈哥,
openFeign异步调用总是失败触发Sentinel的降级,同步调用就没问题?
他还给我晒了一下代码,大致如下:
CompletableFuture<T> future1 = CompletableFuture.supplyAsync(() -> {
//openfeign的调用
return feign.remoteCall();
},executor);
CompletableFuture<T> future2 = CompletableFuture.supplyAsync(() -> {
//openfeign的调用
return feign.remoteCall();
},executor);
CompletableFuture.allOf(future1,future2).join();
.....
这种情况你遇到过吗?
如何解决?
这个算是常见问题了:feign的调用导致上下文丢失;前面有一篇文章说过这种问题:
实战!openFeign如何实现全链路JWT令牌信息不丢失?
在集成OAuth2的时候如果做任何设置会丢失令牌信息,当时我们的解决方案是新建一个拦截器,如下:
@Component
@Slf4j
public class FeignRequestInterceptor implements RequestInterceptor {
@Override
public void apply(RequestTemplate template) {
HttpServletRequest httpServletRequest = RequestContextUtils.getRequest();
Map<String, String> headers = getHeaders(httpServletRequest);
for (Map.Entry<String, String> entry : headers.entrySet()) {
template.header(entry.getKey(), entry.getValue());
* 获取原请求头
private Map<String, String> getHeaders(HttpServletRequest request) {
Map<String, String> map = new LinkedHashMap<>();
Enumeration<String> enumeration = request.getHeaderNames();
if (enumeration != null) {
while (enumeration.hasMoreElements()) {
String key = enumeration.nextElement();
String value = request.getHeader(key);
if (StrUtil.equals(OAuthConstant.TOKEN_NAME,key)){
map.put(key, value);
break;
return map;
}
上述代码根本逻辑就是将请求头中Token信息放入
RequestTemplate
的头中。
“
注意:这里取出请求头中信息用的是
RequestContextHolder
。
”
看到这里是不是明白了,
RequestContextHolder
中是将请求信息放入
ThreadLocal
中的,只能取到同一个线程的数据。
因此要解决异步调用的问题,只需要在发起远程调用之前给异步线程添加上主线程的上下文信息,此时最上方调用失效的代码变成如下:
//获取主线程的请求信息
RequestAttributes attributes = RequestContextHolder.getRequestAttributes();
CompletableFuture<T> future1 = CompletableFuture.supplyAsync(() -> {
//将主线程的请求信息设置到异步线程中,否则会丢失请求上下文,导致调用失败
RequestContextHolder.setRequestAttributes(attributes);
//openfeign的调用
return feign.remoteCall();
},executor);
CompletableFuture<T> future2 = CompletableFuture.supplyAsync(() -> {
//将主线程的请求信息设置到异步线程中,否则会丢失请求上下文,导致调用失败
RequestContextHolder.setRequestAttributes(attributes);
//openfeign的调用
return feign.remoteCall();
},executor);
CompletableFuture.allOf(future1,future2).join();
.....
使用@Async异步注解导致该Bean在循环依赖时启动报BeanCurrentlyInCreationException异常的根本原因分析,以及提供解决方案【享学Spring】(下)
使用@Async异步注解导致该Bean在循环依赖时启动报BeanCurrentlyInCreationException异常的根本原因分析,以及提供解决方案【享学Spring】(上)