Chapter Six《SpringCloud微服务实战》

声明式服务调用: Spring Cloud Feign

Spring Cloud Feign 是什么?

之前有spring cloud ribbonspring cloud hystrix,这二个框架的使用几乎都是同时出现的。是否有更高层次的封装来整合这二个基础工具以简化开发呢?spring cloud feign就是一个这样的工具。它基于Netfix Feign实现,整合了spring cloud Ribbonspring cloud Hystrix,除了提供这二者的强大功能之外,它还提供了一种生命式的web服务客户端定义方式。

spring cloud feign在此基础上做了进一步封装,由他来帮助我们定义和实现依赖服务接口的定义。在spring cloud feign的实现下,我们只需创建一个接口并调用注解的方式来配置它,即可完成对服务提供方的接口绑定,简化了使用spring cloud ribbon时自动封装服务调用客户端的开发量。spring cloud feign具备可插拔的注解支持,包括feign注解和JAX-RS注解。同时,为了适应Spring的广大用户,它在Netfix Feign的基础上扩展了spring mvc的注解支持,这对于习惯spring mvc的开发者来说,无疑是个好消息,因为这样可以大大减少学习使用它的成本。另外,对于feign自身的一些主要组件,比如说编码器和解码器等,它也以可插拔的方式提高,在有需要的时候我们可以方便地扩展和替换它们。

Feign是一种声明式、模板化的HTTP客户端。在Spring Cloud中使用Feign, 我们可以做到使用HTTP请求远程服务时能与调用本地方法一样的编码体验,开发者完全感知不到这是远程方法,更感知不到这是个HTTP请求。

快速入门

创建pay-service服务,加入依赖,与之前的模块不一样的就是加入了spring-cloud-starter-feign依赖,

 <dependencies>
      <dependency>
          <groupId>org.springframework.boot</groupId>
          <artifactId>spring-boot-starter-web</artifactId>
      </dependency>
      <dependency>
          <groupId>org.springframework.boot</groupId>
          <artifactId>spring-boot-starter-test</artifactId>
          <scope>test</scope>
      </dependency>
      <dependency>
          <groupId>org.springframework.cloud</groupId>
          <artifactId>spring-cloud-starter-eureka</artifactId>
      </dependency>
      <dependency>
          <groupId>org.springframework.boot</groupId>
          <artifactId>spring-boot-starter-actuator</artifactId>
      </dependency>
      <dependency>
          <groupId>org.springframework.cloud</groupId>
          <artifactId>spring-cloud-starter-feign</artifactId>
      </dependency>
 </dependencies>

创建主体应用类,并在主体应用类上加上注解@EnableFeignClients:

  @SpringBootApplication
  @EnableDiscoveryClient
  @EnableFeignClients
  public class PayApplication {

        public static void main(String[] args) {
                SpringApplication.run(PayApplication.class,args);
        }
  }

定义UserService接口,通过@FeignClient("user-service")注解指定服务名来绑定服务,然后在使用Spring mvc的注解绑定具体的user-service服务中提供的rest接口。

  @FeignClient("user-service")
      public interface UserService {

  @RequestMapping(value = "/user/index",method = RequestMethod.GET)
      String index();

  @RequestMapping(value = "/user/hello",method = RequestMethod.GET)
      String hello();

接着,创建一个PayController来实现对feign客户端的调用,使用@Autowired注解直接注入上面定义的UserService实例,并在相应的方法中调用这个绑定了user-service服务接口的客户端来向改服务发起接口的定义。

    @RestController
    @RequestMapping("/pay")
    public class PayController {

        private Logger logger = LoggerFactory.getLogger(getClass());

        @Autowired
        UserService userService;

        @RequestMapping("/index")
        public String index(){
            return userService.index();
        }

        @RequestMapping("/hello")
        public String hello(){
            return userService.hello();
        }
    }

最后,同ribbon实现的服务消费一样,需要在application.yml中指定服务注册中心,

    spring:
        application:
          name: pay-service
    eureka:
      client:
        service-url:
         defaultZone: http://localhost:8761/eureka
      instance:
        instance-id:  ${spring.application.name}:${spring.cloud.client.ipAddress}:${spring.application.instance_id:${server.port}}
    server:
      port: 7070

测试验证Ribbon客户端负载均衡,同时启动服务注册和user-servcei,user-service启动了二个实例,然后启动pay-service,


image.png

发送多个请求http://192.168.5.3:7070/pay/index,发现二个user-service都能在控制台打印日志,我们看到了feign实现的消费者,依然是利用了Ribbon维护了user-service的服务列表信息,并且通过轮询实现了客户端的负载均衡。而与Ribbon不同的是,通过feign我们只需要定义服务绑定接口,以声明式的方法,优雅而简单的实现了服务调用。

Spring Cloud的Feign支持的一个中心概念就是命名客户端。 每个Feign客户端都是组合的组件的一部分,它们一起工作以按需调用远程服务器,并且该集合具有您将其作为使用@FeignClient注释的参数名称。 Spring Cloud使用FeignClientsConfiguration创建一个新的集合作为每个命名客户端的ApplicationContext(应用上下文)。 这包含(除其他外)feign.Decoder,feign.Encoder和feign.Contract。
你可以自定义FeignClientsConfiguration以完全控制这一系列的配置。比如我们下面的demo:
定义一个order服务,并加入依赖:

  <dependencies>
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-web</artifactId>
    </dependency>
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-test</artifactId>
        <scope>test</scope>
    </dependency>
    <dependency>
        <groupId>org.springframework.cloud</groupId>
        <artifactId>spring-cloud-starter-eureka</artifactId>
    </dependency>
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-actuator</artifactId>
    </dependency>
    <dependency>
        <groupId>org.springframework.cloud</groupId>
        <artifactId>spring-cloud-starter-feign</artifactId>
    </dependency>
  </dependencies>

定义主体启动类:

  @SpringBootApplication
  @EnableDiscoveryClient
  @EnableFeignClients
  public class OrderApplication {
      public static void main(String[] args) {
          SpringApplication.run(OrderApplication.class,args);
      }
  }

定义Controller:

  @RestController
  @RequestMapping("/order")
  public class OrderController {

      private Logger logger = LoggerFactory.getLogger(getClass());


      @Autowired
      UserService userService;

      @RequestMapping("/index")
      public String index(){
          logger.info("index方法");
          return userService.index();
      }
  }

定义Feign客户端接口:

@FeignClient(value = "user-service",configuration = FooConfiguration.class)
  public interface UserService {

@RequestLine("GET /user/index")
String index();

}

使用了配置@Configuration参数,自己定义了FooConfiguration类来自定义FeignClientsConfiguration,并且FeignClientsConfiguration类的类路径不在启动类OrderApplication的扫描路径下,是因为如果在扫描目录下会覆盖该项目所有的Feign接口的默认配置。
FooConfiguration定义:

  import feign.Contract;
  import org.springframework.context.annotation.Bean;
  import org.springframework.context.annotation.Configuration;

  @Configuration
  public class FooConfiguration {

  //使用Feign自己的注解,使用springmvc的注解就会报错
  @Bean
  public Contract feignContract() {
      return new feign.Contract.Default();
  }
}

配置文件:

spring:
    application:
      name: order-service
  eureka:
    client:
      service-url:
       defaultZone: http://cfy.lessismore:123456@localhost:8761/eureka
instance:
      instance-id:  ${spring.application.name}:${spring.cloud.client.ipAddress}:${spring.application.instance_id:${server.port}}
  server:
      port: 9090

参数绑定

快速入门中,我们使用了spring cloud feign实现的是一个不带参数的REST服务绑定。现实中的各种业务接口要比它复杂的多,我们会在http的各个位置传入各种不同类型的参数,并且在返回请求响应的时候也可能是一个复杂的对象结构。
扩展一下user-servcice服务,增加一些接口定义,其中包含Request参数的请求,带有Header信息的请求,带有RequestBody的请求以及请求响应体中是一个对象的请求,扩展了三个接口分别是hello,hello2,hello3

  @RestController
  @RequestMapping("/user")
  public class UserController {

private final Logger logger = LoggerFactory.getLogger(getClass());

@Autowired
private DiscoveryClient client;

@RequestMapping(value="/index",method = RequestMethod.GET)
public String index(){
    ServiceInstance instance = client.getLocalServiceInstance();
    logger.info("/user,host:"+instance.getHost()+",service id:"+instance.getServiceId()+",port:"+instance.getPort());
    return "user index, local time="+ LocalDateTime.now();
}

@GetMapping("/hello")
public String userHello() throws Exception{
    ServiceInstance serviceInstance = client.getLocalServiceInstance();
    //线程阻塞
    int sleeptime = new Random().nextInt(3000);
    logger.info("sleeptime:"+sleeptime);
    Thread.sleep(sleeptime);
    logger.info("/user,host:"+serviceInstance.getHost()+",service id:"+serviceInstance.getServiceId()+",port:"+serviceInstance.getPort());
    return "user hello";
}

@RequestMapping(value = "/hello1",method = RequestMethod.GET)
public String hello(@RequestParam String username){
    return "hello "+username;
}

@RequestMapping(value = "hello2",method = RequestMethod.GET)
public User hello2(@RequestHeader String username,@RequestHeader Integer age){
    return new User(username,age);
}

@RequestMapping(value = "hello3",method = RequestMethod.POST)
public String hello3(@RequestBody User user){
    return "hello "+user.getUsername() +", "+user.getAge()+", "+user.getId();
}
}

访问:

localhost:8080/user/hello1?username=zhihao.miao

image.png
image.png
image.png

User对象的定义如下,需要注意的是要有User的默认的构造函数,不然,spring cloud feign根据json字符串转换User对象的时候会抛出异常。

public class User {
  private String username;

  private int age;

  private int id;

  public String getUsername() {
      return username;
  }

  public void setUsername(String username) {
      this.username = username;
  }

  public int getAge() {
      return age;
  }

  public void setAge(int age) {
      this.age = age;
  }

  public int getId() {
      return id;
  }

  public void setId(int id) {
      this.id = id;
  }

  public User(String username, int age) {
      this.username = username;
      this.age = age;
  }

  public User() {

  }

  @Override
  public String toString() {
      return "User{" +
              "username='" + username + '\'' +
              ", age=" + age +
              ", id=" + id +
              '}';
  }
}

优点与缺点

使用spring cloud feign的继承特性的优点很明显,可以将接口的定义从Controller 中剥离,同时配合 maven 仓库就可以轻易实现接口定义的共享,实现在构建期的接口绑定,从而有效的减少服务客户端的绑定配置。这么做虽然可以很方便的实现接口定义和依赖的共享,不用在复制粘贴接口进行绑定,但是这样的做法使用不当的话会带来副作用。由于接口在构建期间就建立起了依赖,那么接口变化就会对项目构建造成了影响,可能服务提供方修改一个接口定义,那么会直接导致客户端工程的构建失败。所以,如果开发团队通过此方法来实现接口共享的话,建议在开发评审期间严格遵守面向对象的开闭原则,尽可能低做好前后版本兼容,防止因为版本原因造成接口定义的不一致。

Feign日志的配置

为每个创建的Feign客户端创建一个记录器。默认情况下,记录器的名称是用于创建Feign客户端的接口的完整类名。Feign日志记录仅响应DEBUG级别。logging.level.project.user.UserClient: DEBUG
在配置文件application.yml 中加入:

  logging:
   level:
    com.jalja.org.spring.simple.dao.FeignUserClient: DEBUG 

在自定义的Configuration的类中添加日志级别

  @Configuration
  public class FooConfiguration {
     /* @Bean
      public Contract feignContract() {
        //这将SpringMvc Contract 替换为feign.Contract.Default
          return new feign.Contract.Default();
     }*/
    @Bean
      Logger.Level feignLoggerLevel() {
        //设置日志
        return Logger.Level.FULL;
     }
  }

PS:Feign请求超时问题

Hystrix默认的超时时间是1秒,如果超过这个时间尚未响应,将会进入fallback代码。而首次请求往往会比较慢(因为Spring的懒加载机制,要实例化一些类),这个响应时间可能就大于1秒了
解决方案有三种,以feign为例。

方法一
hystrix.command.default.execution.isolation.thread.timeoutInMilliseconds: 5000
该配置是让Hystrix的超时时间改为5秒
方法二
hystrix.command.default.execution.timeout.enabled: false
该配置,用于禁用Hystrix的超时时间
方法三
feign.hystrix.enabled: false
该配置,用于索性禁用feign的hystrix。该做法除非一些特殊场景,不推荐使用。


Less is more.

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,172评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,346评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,788评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,299评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,409评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,467评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,476评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,262评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,699评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,994评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,167评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,827评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,499评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,149评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,387评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,028评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,055评论 2 352

推荐阅读更多精彩内容