SpringCloud通过Feign实现服务间调用线程变量传输

前言

  在开发项目的过程中,经常需要跨服务传输一些数据,比如将登录的信息从一个服务传递给另一个服务,那么这种情况下需要尽可能少的去动已有的代码,更加体现了编程的封装思想。
  由于SpringCloud并不像dubbo一样,在服务间调用的时候通过本身的线程变量去传输,但是SpringCloud肯定是有它自己的机制可以支持实现的,基于这种考虑,于是有了此篇博客。

实现思路

  1. 准备一个ThreadLoad变量,供线程之间共享
  2. 每个服务对所有过来的Feign调用进行过滤,然后从请求头中获取用户信息,并存在ThreadLocal变量中
  3. 每个服务在使用FeignClient调用其他服务时,先从ThreadLocal变量中取出用户信息,并放在Fegin的request请求头中
  4. 封装一个注解,在启动类中标记即可

关键代码

1、ThreadLocal工具类

public class UserContext {
	//此处线程变量传递的,可以是实体,也可以是某一个变量
    private static ThreadLocal<User> userInfo = new ThreadLocal<User>();
    //此处定义的是要塞到request请求头中的名称
    public static String KEY_USER_IN_HTTP_HEADER = "X-AUTO-FP-USER";

    public UserContext() {
    }

    public static User getUser(){
        return (User)user.get();
    }

    public static void setUser(User user){
        user.set(user);
    }
}

2、用户信息实体类

/**
 * @author 刘子腾
 * @DESCRIPTION 用户信息实体
 * @create 2019/3/28
 */
@Data
public class User {
    /**
     * 用户id
     */
    private String id;
    /**
     * 用户编码
     */
    private String userCode;
    /**
     * 用户名称
     */
    private String userName;
    /**
     * 公司id
     */
    private String companyId;
    /**
     * token
     */
    private String token;
    /**
     * 用户角色id
     */
    private String roleId;
    /**
     *
     * qq登录唯一标示
     */
    private String qqOpenId;
    /**
     *
     * 微信登录唯一标识
     */
    private String weChatOpenId;
    /**
     *
     * 微信公众号唯一标识
     */
    private String wxPlantForm;
}

3、FeignClient拦截器

public class TransmitUserFeighClientIntercepter implements RequestInterceptor {

    private static final Logger log = LoggerFactory.getLogger(TransmitUserFeighClientIntercepter.class);
    public TransmitUserFeighClientIntercepter() {
    }

    @Override
    public void apply(RequestTemplate requestTemplate) {
        //从应用上下文中取出user信息,放入Feign的请求头中
        User user = UserContext.getUser();
        if (user != null) {
            try {
                String userJson = JSON.toJSONString(user);
                requestTemplate.header("KEY_USER_IN_HTTP_HEADER",new String[]{URLDecoder.decode(userJson,"UTF-8")});
            } catch (UnsupportedEncodingException e) {
                log.error("用户信息设置错误",e);
            }
        }
    }
}

4、Filter过滤器

public class TransmitUserFilter implements Filter {

    private static final Logger log = LoggerFactory.getLogger(TransmitUserFeighClientIntercepter.class);
    public TransmitUserFilter() {
    }
    @Override
    public void init(FilterConfig filterConfig) throws ServletException {
    }
    @Override
    public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {
       this.initUserInfo((HttpServletRequest)request);
       chain.doFilter(request,response);
    }

    private void initUserInfo(HttpServletRequest request){
        String userJson = request.getHeader("KEY_USER_IN_HTTP_HEADER");
        if (StringUtils.isNotBlank(userJson)) {
            try {
                userJson = URLDecoder.decode(userJson,"UTF-8");
                User userInfo = (User) JSON.parseObject(userJson,UserInfo.class);
                //将用户信息放入上下文中
                UserContext.setUser(user);
            } catch (UnsupportedEncodingException e) {
               log.error("init userInfo error",e);
            }
        }
    }

    @Override
    public void destroy() {
    }
}

5、自定义注解

@Documented
@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Import({EnableUserTransmitterAutoConfiguration.class})
public @interface EnableUserTransmitter {
}

6、自定义注解实现类

@Configuration
public class EnableUserTransmitterAutoConfiguration {

    public EnableUserTransmitterAutoConfiguration() {
    }

    @Bean
    public TransmitUserFeighClientIntercepter transmitUser2FeighHttpHeader(){
       return new TransmitUserFeighClientIntercepter();
    }

    @Bean
    public TransmitUserFilter transmitUserFromHttpHeader(){
        return new TransmitUserFilter();
    }
}

7、在启动类上标记自定义注解

### 如何在 Feign 客户端调用远程服务时设置自定义请求头 为了确保 Feign 调用能够携带特定的请求头信息,可以通过 `RequestInterceptor` 接口实现这一功能。每当 Feign 发送 HTTP 请求时,会自动触发该拦截器中的逻辑。 #### 使用 RequestInterceptor 添加请求头 创建一个实现了 `RequestInterceptor` 的类,在其中编写具体的业务逻辑: ```java import feign.RequestInterceptor; import feign.RequestTemplate; public class CustomFeignInterceptor implements RequestInterceptor { @Override public void apply(RequestTemplate template) { // 设置固定的请求头 template.header("X-Custom-Header", "CustomValue"); // 或者动态获取并添加请求头 String token = getTokenFromSomewhere(); if (token != null && !"".equals(token.trim())) { template.header("Authorization", "Bearer " + token); } } } ``` 为了让上述拦截器生效,还需要将其注册到 Spring 上下文中。可以在配置文件中声明 bean 实例: ```yaml @Configuration public class FeignConfig { @Bean public RequestInterceptor requestInterceptor() { return new CustomFeignInterceptor(); } } ``` 如果希望某个 Feign Client 特定情况下应用不同的配置,则可在 `@FeignClient` 注解内指定对应的配置类[^4]。 对于更复杂的场景,比如传递当前线程上下文中的认证信息或其他环境变量,也可以利用 `RequestContextHolder` 来访问原始请求的数据,并据此调整即将发出的新请求的内容[^1]。
评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值