@FeignClient中的@RequestMapping也被SpringMVC加载的问题解决

问题描述

在之前发布的《Spring Cloud实战小贴士:Feign的继承特性(伪RPC模式)》一文中,我们介绍了如果使用Feign的继承特性来完成服务的提供以及服务的消费,实现了类似RPC的编程模式。但是,仔细一些的读者可能已经发现一个问题:当我们将服务消费者运行起来的时候,定义在服务提供方的那些请求映射关系也被加载到了服务消费者中,这就会带来两个问题:

由于服务消费者并不提供这些接口,对于开发者来说容易造成误解
由于加载了一些外部服务的接口定义,还存在与自身接口定义冲突的潜在风险
问题分析
那么这些外部请求接口定义是如何被加载到消费端的呢?我们先来看看Spring MVC处理请求映射的RequestMappingHandlerMapping实现片段:

@Override
protected boolean isHandler(Class<?> beanType) {
	return (AnnotatedElementUtils.hasAnnotation(beanType, Controller.class) ||
			AnnotatedElementUtils.hasAnnotation(beanType, RequestMapping.class));
}

我们可以发现如上的这段实现,该函数用来判断是否要处理请求映射的判断依据。从实现中我们看到,只要被扫描的类包含了@Controller注解或@RequestMapping注解,那么就会被加载进来。虽然@FeignClient定义修饰的服务消费端没有声明这些注解,但是当我们使用了继承特性的时候,那么这些注解就也会被服务消费者解析和加载,所以出现了上面所描述的现象。

解决方法

既然已经找到了问题所在,那么我们可以针对性的扩展处理:扩展RequestMappingHandlerMapping的isHandler函数。

@Configuration
@ConditionalOnClass({Feign.class})
public class FeignConfiguration {

    @Bean
    public WebMvcRegistrations feignWebRegistrations() {
        return new WebMvcRegistrationsAdapter() {
            @Override
            public RequestMappingHandlerMapping getRequestMappingHandlerMapping() {
                return new FeignRequestMappingHandlerMapping();
            }
        };
    }

    private static class FeignRequestMappingHandlerMapping extends RequestMappingHandlerMapping {
        @Override
        protected boolean isHandler(Class<?> beanType) {
            return super.isHandler(beanType) &&
                    !AnnotatedElementUtils.hasAnnotation(beanType, FeignClient.class);
        }
    }
}

如上实现的isHandler函数继承了原来的实现,同时增加了一个条件:不能被@FeignClient注解修饰的类才会进行解析加载。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
@FeignClient和@RequestMapping是Spring框架常用的注解,用于实现微服务之间的通信和URL映射。 @FeignClient注解用于声明一个Feign客户端接口,用于调用其他微服务的API。它需要指定要调用的微服务的名称,可以通过value或name属性指定。Feign客户端会根据接口定义自动创建代理对象,简化了与其他微服务的交互。 例如,假设有一个名为UserService的微服务,我们可以使用@FeignClient注解来声明一个与UserService交互的Feign客户端接口: ```java @FeignClient(name = "user-service") public interface UserFeignClient { @GetMapping("/users/{id}") User getUserById(@PathVariable("id") Long id); } ``` 在上面的例子,我们声明了一个名为UserFeignClientFeign客户端接口,它与名为"user-service"的微服务进行通信。接口定义了一个getUserById方法,用于调用"user-service"提供的获取用户信息的API。 @RequestMapping注解用于将HTTP请求映射到控制器方法上。它可以用于类级别和方法级别。在类级别上使用@RequestMapping注解可以定义共享的请求前缀,而在方法级别上使用可以定义具体的请求路径和HTTP方法。 例如,我们可以在控制器类上使用@RequestMapping注解来定义请求前缀: ```java @RestController @RequestMapping("/api/users") public class UserController { // ... } ``` 在上面的例子,所有以"/api/users"开头的请求都会被映射到UserController类的方法上。 在方法级别上使用@RequestMapping注解可以指定具体的请求路径和HTTP方法。例如: ```java @RestController @RequestMapping("/api/users") public class UserController { @GetMapping("/{id}") public User getUserById(@PathVariable("id") Long id) { // ... } } ``` 在上面的例子,GET请求"/api/users/{id}"会被映射到getUserById方法上,其{id}是一个路径变量,用于指定要获取的用户的ID。 总结一下,@FeignClient用于声明Feign客户端接口,用于调用其他微服务的API;@RequestMapping用于将HTTP请求映射到控制器方法上,可以定义请求前缀、请求路径和HTTP方法。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值