负载均衡Ribbon&Feign

1.1. 什么是Ribbon负载均衡

Ribbon是一个基于客户端的负载均衡框架,它主要用于在服务消费者和服务提供者之间进行负载均衡。Ribbon具有以下特点:

  1. 基于客户端的负载均衡:Ribbon是一种基于客户端的负载均衡框架,它将负载均衡的逻辑集成到服务消费者中,通过客户端自己进行负载均衡,避免了服务提供者对负载均衡的依赖,使得系统更加灵活和可靠。
  2. 支持多种负载均衡算法:Ribbon提供了多种负载均衡算法,例如轮询、随机、权重等,可以根据不同的业务需求选择不同的负载均衡算法。
  3. 支持服务发现:Ribbon支持服务发现,可以自动发现服务提供者的地址和端口号,无需手动配置。它可以与Eureka、Consul等服务注册中心进行集成,实现自动化的服务发现和负载均衡。
  4. 支持自定义配置:Ribbon支持自定义配置,可以根据不同的业务场景进行配置,例如超时时间、重试次数、连接池大小等。
  5. 集成Spring Cloud:Ribbon是Spring Cloud中默认的负载均衡框架,可以与Spring Cloud无缝集成,实现服务之间的负载均衡。

总之,Ribbon是一个非常优秀的基于客户端的负载均衡框架,它可以有效地提高系统的可靠性和性能,同时也非常灵活和易于配置。在微服务架构中,Ribbon可以与Eureka、Consul等服务注册中心和Spring Cloud等微服务框架无缝集成,实现自动化的服务发现和负载均衡。




openFeign底层负载均衡_ribbon


1.2. 自定义实现负载均衡

  • ribbon的启动器

nacos已经集成了ribbon的依赖,故无需添加启动器

1.2.1. 创建服务提供者

创建子模块工程ribbon_provider_1

pom

<?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">
    <parent>
        <artifactId>springcloud-parent</artifactId>
        <groupId>cloud.funcode</groupId>
        <version>1.0-SNAPSHOT</version>
    </parent>
    <modelVersion>4.0.0</modelVersion>

    <artifactId>ribbon_provider_1</artifactId>

    <properties>
        <maven.compiler.source>8</maven.compiler.source>
        <maven.compiler.target>8</maven.compiler.target>
    </properties>

    <dependencies>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>
        <dependency>
            <groupId>cloud.funcode</groupId>
            <artifactId>springcloud_common</artifactId>
            <version>1.0-SNAPSHOT</version>
        </dependency>

        <dependency>
            <groupId>com.alibaba.cloud</groupId>
            <artifactId>spring-cloud-starter-alibaba-nacos-discovery</artifactId>
        </dependency>
    </dependencies>

</project>

yml配置文件

server:
  port: 8090
spring:
  cloud:
    nacos:
      discovery:
        server-addr: 192.168.159.129:8848 #nacos虚拟机地址+端口号
        namespace: dev
        group: xx
  application:
    name: ribbon-provider

业务层接口层TestService及其实现层TestServiceImpl

public interface TestService {

    public User getUserById(Integer id);
}
@Service
public class TestServiceImpl implements TestService {
    @Override
    public User getUserById(Integer id) {
        return new User(id,"范特西",25);
    }
}

控制层Controller

@RestController
@RequestMapping("/provider")
public class TestController {
    @Autowired
    private TestService service;

    @GetMapping("/getUserById/{id}")
    public User getUserById(@PathVariable("id") Integer id){
        return service.getUserById(id);
    }
}

重复上述步骤 再创建一个子模块工程ribbon_provider_2

在父项目【版本控制模块】的 pom.xml中引入两个子模块

<modules>
        <module>springcloud_common</module>
        <module>nacos_provider</module>
        <module>nacos_consumer</module>
        <module>nacos_config</module>
        <module>ribbon_provider_1</module>
        <module>ribbon_provider_2</module>
    </modules>

1.2.2. 创建服务消费者

application.yml

server:
  port: 80
spring:
  cloud:
    nacos:
      discovery:
        server-addr: 192.168.209.129:8848
  application:
    name: ribbon-consumer

控制层controller

@RestController
@RequestMapping(value = "/consumer")
public class ConsumerController {
	
	@Autowired
	private RestTemplate restTemplate;

	@Autowired
	private DiscoveryClient discoveryClient;

	private int currentIndex;

	@RequestMapping(value="/getUserById/{id}")
	public User getUserById(@PathVariable Integer id){
		List<ServiceInstance> serviceList = 
            				discoveryClient.getInstances("ribbon-provider");

		//随机方式获得服务
		//int currentIndex = new Random().nextInt(serviceList.size());
		//轮询方式获得服务
		currentIndex = (currentIndex + 1) % serviceList.size();
		
		ServiceInstance instance = serviceList.get(currentIndex);
		String serviceUrl = instance.getHost() + ":" + instance.getPort();
        System.out.println("serviceUrl:"+serviceUrl);
		String url = "http://"+serviceUrl+"/provider/getUserById/"+id;
		return restTemplate.getForObject(url, User.class);
	}
}

BeanConfig

@Configuration
public class BeanConfig {

    @LoadBalanced //开启ribbon负载均衡
    @Bean
    public RestTemplate restTemplate(){
        return new RestTemplate();
    }
    
    @Bean
    public IRule iRule(){
        return new RandomRule();随机策略
    }
}

在父项目【版本控制模块】的 pom.xml中引入ribbon_consumer模块

<modules>
        <module>springcloud_common</module>
        <module>nacos_provider</module>
        <module>nacos_consumer</module>
        <module>nacos_config</module>
        <module>ribbon_provider_1</module>
        <module>ribbon_provider_2</module>
        <module>ribbon_consumer</module>
    </modules>

1.2.3. 测试

  • 启动虚拟机Nacos服务

浏览器地址栏输入:http://192.168.159.129:8848/nacos,出现以下界面代表启动成功


openFeign底层负载均衡_Powered by 金山文档_02


  • 然后idea中启动以下三个模块


openFeign底层负载均衡_spring cloud_03


浏览器地址栏输入:http://127.0.0.1/consumer/getUserById/520,出现以下界面代表配置成功


openFeign底层负载均衡_负载均衡_04


经过上述配置仍存在些许问题,比如拼接url和参数显得好傻,所以接下来要使用feign 处理参数拼接问题

1.3. 什么是feign

Feign是一个轻量级的HTTP客户端,它使用了基于注解的方式来定义和描述RESTful服务的接口,使得客户端代码变得更加简洁和易于维护。它的特点包括:

  1. 与Spring Cloud集成:Feign可以与Spring Cloud无缝集成,使得服务之间的通信更加简单和可靠。它可以自动发现服务并进行负载均衡,同时也支持服务降级和熔断等机制,保证了服务的高可用性和可靠性。
  2. 声明式的API:Feign使用基于注解的方式来定义和描述RESTful服务的接口,它提供了一种声明式的API,使得代码更加易于理解和维护。通过简单的注解,可以定义接口的路径、请求方式、请求参数和返回类型等信息,而无需关注具体的HTTP请求和响应细节。
  3. 易于扩展:Feign提供了许多自定义的扩展点,使得用户可以根据自己的需求来扩展和定制Feign的功能。例如,可以自定义编码器和解码器、请求拦截器、日志记录器等,从而满足各种不同的业务场景。
  4. 支持多种HTTP客户端:Feign默认使用的是基于Java的HTTP客户端,但也支持其他HTTP客户端的集成,例如Apache HttpClient、OkHttp等,从而使得Feign可以更加灵活地适应各种不同的应用场景。

总之,Feign是一个非常优秀的Java HTTP客户端,它的设计理念简单、易用、可扩展,可以大大提高开发效率和代码可维护性。

注意:

  • feign[feɪn]是springcloud提供的声名式(接口)的http客户端(工作在consumer端)
  • feign支持springmvc注解
  • feign集成了ribbon也支持负载均衡
  • feign = ribbon + restTemplate

1.3.1. 创建feign_interface

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">
    <parent>
        <artifactId>springcloud-parent</artifactId>
        <groupId>cloud.funcode</groupId>
        <version>1.0-SNAPSHOT</version>
    </parent>
    <modelVersion>4.0.0</modelVersion>

    <artifactId>feign_interface</artifactId>

    <properties>
        <maven.compiler.source>8</maven.compiler.source>
        <maven.compiler.target>8</maven.compiler.target>
    </properties>

    <dependencies>
        <dependency>
            <groupId>cloud.funcode</groupId>
            <artifactId>springcloud_common</artifactId>
            <version>1.0-SNAPSHOT</version>
        </dependency>

        <dependency>
            <groupId>org.springframework.cloud</groupId>
            <artifactId>spring-cloud-starter-openfeign</artifactId>
        </dependency>
    </dependencies>

</project>

创建UserFeign接口

@FeignClient("feign-provider")//代理类
@RequestMapping("/provider")
public interface UserFeign {

    @RequestMapping("/getUserById/{id}") //拼接url
    public User getUserById(@PathVariable("id") Integer id);//@PathVariable拼接参数,?"id"
}

1.3.2. 创建feign_consumer

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">
    <parent>
        <artifactId>springcloud-parent</artifactId>
        <groupId>cloud.funcode</groupId>
        <version>1.0-SNAPSHOT</version>
    </parent>
    <modelVersion>4.0.0</modelVersion>

    <artifactId>feign_consumer</artifactId>

    <properties>
        <maven.compiler.source>8</maven.compiler.source>
        <maven.compiler.target>8</maven.compiler.target>
    </properties>

    <dependencies>
        <dependency>
            <groupId>cloud.funcode</groupId>
            <artifactId>springcloud_common</artifactId>
            <version>1.0-SNAPSHOT</version>
        </dependency>

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>

        <dependency>
            <groupId>com.alibaba.cloud</groupId>
            <artifactId>spring-cloud-starter-alibaba-nacos-discovery</artifactId>
        </dependency>

        <dependency>
            <groupId>cloud.funcode</groupId>
            <artifactId>feign_interface</artifactId>
            <version>1.0-SNAPSHOT</version>
        </dependency>
    </dependencies>

</project>

控制层TestController

@RestController
@RequestMapping("/consumer")
public class TestController {
    @Autowired
    private UserFeign userFeign;

    @GetMapping("/getUserById/{id}")
    public User getUserById(@PathVariable Integer id) {
        return userFeign.getUserById(id);
    }
}

application.yml

server:
  port: 80
spring:
  cloud:
    nacos:
      discovery:
        server-addr: 192.168.159.129:8848
  application:
    name: feign-consumer

启动器类

@SpringBootApplication
@EnableDiscoveryClient
@EnableFeignClients//开启feign
public class FeignConsumerApp {
    public static void main(String[] args) {
        SpringApplication.run(FeignConsumerApp.class, args);
    }
}

1.3.3. 创建feign_provider

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">
    <parent>
        <artifactId>springcloud-parent</artifactId>
        <groupId>cloud.funcode</groupId>
        <version>1.0-SNAPSHOT</version>
    </parent>
    <modelVersion>4.0.0</modelVersion>

    <artifactId>feign_provider</artifactId>

    <properties>
        <maven.compiler.source>8</maven.compiler.source>
        <maven.compiler.target>8</maven.compiler.target>
    </properties>

    <dependencies>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>
        <dependency>
            <groupId>cloud.funcode</groupId>
            <artifactId>springcloud_common</artifactId>
            <version>1.0-SNAPSHOT</version>
        </dependency>

        <dependency>
            <groupId>com.alibaba.cloud</groupId>
            <artifactId>spring-cloud-starter-alibaba-nacos-discovery</artifactId>
        </dependency>
    </dependencies>

</project>

application.yaml

server:
  port: 8090
spring:
  cloud:
    nacos:
      discovery:
        server-addr: 192.168.159.129:8848 #nacos虚拟机地址+端口号
  application:
    name: feign-provider #和工程名保持一致

控制层Controller

@RestController
@RequestMapping("/provider")
public class TestController {
    @Autowired
    private TestService service;

    @GetMapping("/getUserById/{id}")
    public User getUserById(@PathVariable("id") Integer id){
        return service.getUserById(id);
    }
}

业务层Service以及ServiceImpl

public interface TestService {

    public User getUserById(Integer id);
}
@Service
public class TestServiceImpl implements TestService {
    @Override
    public User getUserById(Integer id) {
        return new User(id,"范特西",25);
    }
}

启动器

@SpringBootApplication
@EnableDiscoveryClient//开启nacos的功能:注册自己并发现其他服务
public class FeignProviderApp {
    public static void main(String[] args) {
        SpringApplication.run(FeignProviderApp.class, args);
    }
}

在父项目【版本控制模块】的 pom.xml中引入以上模块

<modules>
        <module>springcloud_common</module>
        <module>nacos_provider</module>
        <module>nacos_consumer</module>
        <module>nacos_config</module>
        <module>ribbon_provider_1</module>
        <module>ribbon_provider_2</module>
        <module>ribbon_consumer</module>
        <module>feign_interface</module>
        <module>feign_consumer</module>
        <module>feign_provider</module>
    </modules>

1.3.4. 测试

浏览器地址栏输入:http://127.0.0.1/consumer/getUserById/520,出现以下界面代表配置成功


openFeign底层负载均衡_负载均衡_05