SpringCloud---(6)Feign(声明式服务调用)

Feign是一个声明式、模板化的HTTP客户端,它使得HTTP客户端变得更简单。使用Feign,只需要创建一个接口并用注解方式来配置它,即可完成对服务提供者的接口绑定,简化了在使用Ribbon时自行封装服务调用客户端的开发量。

Feign具有可插拔的注解特性,包括Feign注解和JAX-RS注解,同时也扩展了对Spring MVC的注解支持。Feign支持可插拔的编码器的解码器,默认集成了Ribbon,并和Eureka结合,默认实现了负载均衡的效果。

在pom中增加spring-cloud-starter-feign依赖包

 

1 Feign的基础应用

1.1 pom.xml

注:引入依赖

<?xml version="1.0" encoding="UTF-8"?>

<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"

xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">

<modelVersion>4.0.0</modelVersion>



<artifactId>microservice-consumer-movie-feign</artifactId>

<packaging>jar</packaging>



<parent>

    <groupId>com.kevin.cloud</groupId>

    <artifactId>microservice-spring-cloud</artifactId>

    <version>0.0.1-SNAPSHOT</version>

    <relativePa
th/>

</parent>


<properties>

    <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>

    <project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding>

    <java.version>1.8</java.version>

</properties>


<dependencies>

    <dependency>

    <groupId>org.springframework.boot</groupId>

    <artifactId>spring-boot-starter-web</artifactId>

</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>


</project>

1.2 application.yml

server:

  port: 7901

spring:

  application:

    name: microservice-consumer-movie

eureka:

  client:

    healthcheck:

      enabled: true

service-url:

      defaultZone: http://kevin:123456@localhost:8761/eureka

instance:

    prefer-ip-address: true

 

1.3 ConsumerMovieFeignApplication.java

注:在springboot主类加入注解,@EnableFeignClients开启Feign功能。

package com.keivn.cloud;

import org.springframework.boot.SpringApplication;

import org.springframework.boot.autoconfigure.SpringBootApplication;

import org.springframework.cloud.netflix.eureka.EnableEurekaClient;

import org.springframework.cloud.netflix.feign.EnableFeignClients;


/**

 *

 * @title   服务消费者

 * @description

* @author caonanqing

 * @createDate 2018/11/7

 * @version 1.0

 */

@SpringBootApplication

@EnableEurekaClient    //注册到Eureka

@EnableFeignClients    //开启Fegin功能

public class ConsumerMovieFeignApplication {



public static void main(String[] args) {

      SpringApplication.run(ConsumerMovieFeignApplication.class, args);

      System.out.println("服务消费者启动...");

   }

}

1.4 UserFeignClient.java

注:注入接口,@FeignClient("microservice-provider-user") //绑定该接口对应的服务,不能使用GETMapping,需要使用RequestMapping
 

package com.keivn.cloud.feign;

import com.keivn.cloud.entity.User;

import org.springframework.cloud.netflix.feign.FeignClient;

import org.springframework.web.bind.annotation.*;


/**

 *

 * @title

* @description

* @author caonanqing

 * @createDate 2018/11/9

 * @version 1.0

 */

@FeignClient("microservice-provider-user") //绑定该接口对应的服务

public interface UserFeignClient {

// 1.@GetMapping不支持,2.@PathVarible必须要得设置value

@RequestMapping(value = "/simple/{id}", method = RequestMethod.GET)

public User findById(@PathVariable("id") Long id );

// 测试POST请求方式

@RequestMapping(value = "/post-user", method = RequestMethod.POST)

public User postUser(@RequestBody User user);

// 测试GET请求方式

@RequestMapping(value = "/get-user", method = RequestMethod.GET)

public User getUser(User user);

}

1.5 MovieController.java

注:在Controller类调用接口

package com.keivn.cloud.controller;

import com.keivn.cloud.feign.UserFeignClient;

import com.keivn.cloud.entity.User;

import org.springframework.beans.factory.annotation.Autowired;

import org.springframework.web.bind.annotation.*;

/**

 *

 * @title

* @description

* @author caonanqing

 * @createDate 2018/11/7

 * @version 1.0

 */

@RestController

public class MovieController {

@Autowired

private UserFeignClient userFeignClient;

@GetMapping(value = "/movie/{id}")

public User finById(@PathVariable long id ){

return this.userFeignClient.findById(id);

    }

@GetMapping(value = "/post-user")

public User postUser(User user){

return this.userFeignClient.postUser(user);

    }

@GetMapping(value = "/get-user")

public User getUser(User user){

return this.userFeignClient.getUser(user);

    }

}

1.6 User.java

package com.keivn.cloud.entity;



import java.io.Serializable;

import java.math.BigDecimal;



/**

 *

 * @title

* @description

* @author caonanqing

 * @createDate 2018/11/7

 * @version 1.0

 */

public class User implements Serializable {



private Long id;



private String username;



private String name;



private Short age;



private BigDecimal balance;



public User() {

    }



public User(Long id, String username, String name, Short age, BigDecimal balance) {

this.id = id;

this.username = username;

this.name = name;

this.balance = balance;

    }



public Long getId() {

return id;

    }



public void setId(Long id) {

this.id = id;

    }



public String getUsername() {

return username;

    }



public void setUsername(String username) {

this.username = username;

    }



public String getName() {

return name;

    }



public void setName(String name) {

this.name = name;

    }



public Short getAge() {

return age;

    }



public void setAge(Short age) {

this.age = age;

    }



public BigDecimal getBalance() {

return balance;

    }



public void setBalance(BigDecimal balance) {

this.balance = balance;

    }

}

 

2 Feign覆写Feign的默认配置及日志

Spring Cloud 的Feign支持的一个中心概念是命名客户端。

每个Feign Client是整体的一部分,它们一起工作以按需联系远程服务器,并且该整体具有一个名称,开发人员可以使用@FeignClient将其命名。

SpringCloud根据需要使用FeignClientsConfiguration为每个命名的客户端创建一个新的整体作为ApplicationContext。这包含(其他)Feign.Docoder,Feign.Encoder和Feign.Contract。

 

通过额外的配置来控制Feign Client(覆写默认配置)

pom与上面1.1的一致,不需要改动,user类与上面的也是一致

2.1 application.yml

server:

  port: 7901

spring:

  application:

    name: microservice-consumer-movie

eureka:

  client:

    healthcheck:

      enabled: true

service-url:

      defaultZone: http://kevin:123456@localhost:8761/eureka

instance:

    prefer-ip-address: true

logging:  #启动日志

level:

    com.keivn.cloud.feign.UserFeignClient: debug



# 解决第一次请求报超时异常的方案:

# hystrix.command.default.execution.isolation.thread.timeoutInMilliseconds: 5000

# 或者:

# hystrix.command.default.execution.timeout.enabled: false

# 或者:

feign.hystrix.enabled: false  # 索性禁用feign的hystrix



# 超时的issue:https://github.com/spring-cloud/spring-cloud-netflix/issues/768

# 超时的解决方案: http://stackoverflow.com/questions/27375557/hystrix-command-fails-with-timed-out-and-no-fallback-available

# hystrix配置: https://github.com/Netflix/Hystrix/wiki/Configuration#execution.isolation.thread.timeoutInMilliseconds

2.2 ConsumerMovieFeignApplication.java

package com.keivn.cloud;

import org.springframework.boot.SpringApplication;

import org.springframework.boot.autoconfigure.SpringBootApplication;

import org.springframework.cloud.netflix.eureka.EnableEurekaClient;

import org.springframework.cloud.netflix.feign.EnableFeignClients;



/**

 *

 * @title   服务消费者

 * @description

* @author caonanqing

 * @createDate 2018/11/7

 * @version 1.0

 */

@SpringBootApplication

@EnableEurekaClient    //注册到Eureka

@EnableFeignClients    //开启Fegin功能

public class ConsumerMovieFeignApplication {



public static void main(String[] args) {

      SpringApplication.run(ConsumerMovieFeignApplication.class, args);

      System.out.println("服务消费者启动...");

   }

}

2.3 UserFeignClient.java

package com.keivn.cloud.feign;

import com.keivn.cloud.entity.User;

import com.keivn.config.Configuration1;

import feign.Param;

import feign.RequestLine;

import org.springframework.cloud.netflix.feign.FeignClient;



/**

 *

 * @title

* @description

* @author caonanqing

 * @createDate 2018/11/9

 * @version 1.0

 */

@FeignClient(name = "microservice-provider-user", configuration = Configuration1.class) //绑定该接口对应的服务,以及配置

public interface UserFeignClient {



// 覆写Feign默认的配置,使用的Feign本身的注解,而不是SpringMVC

@RequestLine("GET /simple/{id}") //GET请求+空格+请求地址

public User findById(@Param("id") Long id );



}

2.4 UserFeignClient2.java

package com.keivn.cloud.feign;

import com.keivn.config.Configuration2;

import org.springframework.cloud.netflix.feign.FeignClient;

import org.springframework.web.bind.annotation.PathVariable;

import org.springframework.web.bind.annotation.RequestMapping;


/**

 *

 * @title

* @description

* @author caonanqing

 * @createDate 2018/11/14

 * @version 1.0

 */

//xxx只是随意起的名称,实际是Eureka中Application的名称

@FeignClient(name = "xxx", url = "http://localhost:8761/", configuration = Configuration2.class) //绑定该接口对应的服务,以及配置

public interface UserFeignClient2 {

//访问eureka里注册的服务的名称

@RequestMapping(value ="/eureka/apps/{serviceName}")

public String findServiceInfoFromEurekaByServiceName(@PathVariable("serviceName") String serviceName);


}

2.5 Configuration1.java

package com.keivn.config;

import feign.Contract;

import feign.Logger;

import org.springframework.context.annotation.Configuration;

import org.springframework.context.annotation.Bean;



/**

 *

 * @title   通过声明额外的配置来控制Feign Client(覆写默认配置)

 * @description 为了不与Spring Boot的Application的ComponentScan包冲突,将配置类放到cloud包外面

 * @author caonanqing

 * @createDate 2018/11/14

 * @version 1.0

 */

@Configuration

public class Configuration1 {



// 契约,Contract使用的是Feign本身的注解

@Bean

public Contract feignContract(){

return new feign.Contract.Default();

    }



// 启动日志

@Bean

Logger.Level feignLoggerLever(){

return Logger.Level.FULL;

    }



}

2.6 Configuration2.java

package com.keivn.config;

import feign.Contract;

import feign.auth.BasicAuthRequestInterceptor;

import org.springframework.context.annotation.Bean;

import org.springframework.context.annotation.Configuration;



/**

 *

 * @title

* @description

* @author caonanqing

 * @createDate 2018/11/14

 * @version 1.0

 */

@Configuration

public class Configuration2 {



// 直接访问是没有权限的,需要加上该注解配置服务提供者的账号密码。

@Bean

public BasicAuthRequestInterceptor basicAuthRequestInterceptor(){

return new BasicAuthRequestInterceptor("kevin","123456");

    }



}

2.7 MovieController.java

package com.keivn.cloud.controller;

import com.keivn.cloud.feign.UserFeignClient;

import com.keivn.cloud.entity.User;

import com.keivn.cloud.feign.UserFeignClient2;

import org.springframework.beans.factory.annotation.Autowired;

import org.springframework.web.bind.annotation.*;



/**

 *

 * @title

* @description

* @author caonanqing

 * @createDate 2018/11/7

 * @version 1.0

 */

@RestController

public class MovieController {



@Autowired

private UserFeignClient userFeignClient;



@Autowired

private UserFeignClient2 userFeignClient2;



@GetMapping(value = "/movie/{id}")

public User finById(@PathVariable long id ){

return this.userFeignClient.findById(id);

    }



@GetMapping("/{serviceName}")

public String findServiceInfoFromEurekaByServiceName(@PathVariable String serviceName){

return this.userFeignClient2.findServiceInfoFromEurekaByServiceName(serviceName);

    }

}

 

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值