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

问题描述

在之前发布的《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定义修饰的服务消费端没有声明这些注解,但是当我们使用了继承特性的时候,那么这些注解就也会被服务消费者解析和加载,所以出现了上面所描述的现象。

解决方法

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

@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注解修饰的类才会进行解析加载。

原文发布于微信公众号 - 程序猿DD(didispace)

原文发表时间:2017-08-13

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏pangguoming

Spring Boot Maven Plugin打包异常及三种解决方法:Unable to find main class

5732
来自专栏帅小子的日常

使用redis做缓存

8037
来自专栏程序员互动联盟

linux设备驱动第五篇:驱动中的并发与竟态

综述 在上一篇介绍了linux驱动的调试方法,这一篇介绍一下在驱动编程中会遇到的并发和竟态以及如何处理并发和竞争。 首先什么是并发与竟态呢?并发(concurr...

35810
来自专栏xingoo, 一个梦想做发明家的程序员

Elasticsearch安装

在启动或者安装ES之前,需要先下载JDK 1.7以上的版本,对于2.0来说,要求JDK1.8以上。 检查JDK的版本 使用命令: java -versio...

2546
来自专栏A周立SpringCloud

如何使用Feign构造多参数的请求

最近经常有人在Spring Cloud中国社区(http://springcloud.cn)QQ群(157525002)里问到该问题。索性整理一下。 本节我们来...

5555
来自专栏Golang语言社区

linux 内核同步机制使用

Linux 内核中的同步机制:原子操作、信号量、读写信号量、自旋锁的API、大内核锁、读写锁、大读者锁、RCU和顺序锁。 1、介绍 在现代操作系统里,同一时间...

3995
来自专栏Java修行之道

SpringMVC中controller接收Json数据

2291
来自专栏JavaEE

Thymeleaf的使用前言:一、thymeleaf简介:二、thymeleaf标准方言:三、thymeleaf与springboot集成案例:总结:

最近听说thymeleaf好像也挺流行的,还说是spring官方推荐使用,那thymeleaf究竟是什么呢?spring为什么推荐用它呢?怎么用呢?本文将为你揭...

1432
来自专栏雅俗

$.ajax使用Form提交与Payload提交

Form解析可以直接从Request对象中获取请求参数,这样对象转换与处理相对容易,但在大批JSON数据需要提交时,可能会出现大量的数据拆分与处理工作,另外针对...

1608
来自专栏我是攻城师

在spring-boot中使用@ConfigurationProperties注解

6644

扫码关注云+社区

领取腾讯云代金券