负载均衡Ribbon&Feign
1.1. 什么是Ribbon负载均衡
Ribbon是一个基于客户端的负载均衡框架,它主要用于在服务消费者和服务提供者之间进行负载均衡。Ribbon具有以下特点:
- 基于客户端的负载均衡:Ribbon是一种基于客户端的负载均衡框架,它将负载均衡的逻辑集成到服务消费者中,通过客户端自己进行负载均衡,避免了服务提供者对负载均衡的依赖,使得系统更加灵活和可靠。
- 支持多种负载均衡算法:Ribbon提供了多种负载均衡算法,例如轮询、随机、权重等,可以根据不同的业务需求选择不同的负载均衡算法。
- 支持服务发现:Ribbon支持服务发现,可以自动发现服务提供者的地址和端口号,无需手动配置。它可以与Eureka、Consul等服务注册中心进行集成,实现自动化的服务发现和负载均衡。
- 支持自定义配置:Ribbon支持自定义配置,可以根据不同的业务场景进行配置,例如超时时间、重试次数、连接池大小等。
- 集成Spring Cloud:Ribbon是Spring Cloud中默认的负载均衡框架,可以与Spring Cloud无缝集成,实现服务之间的负载均衡。
总之,Ribbon是一个非常优秀的基于客户端的负载均衡框架,它可以有效地提高系统的可靠性和性能,同时也非常灵活和易于配置。在微服务架构中,Ribbon可以与Eureka、Consul等服务注册中心和Spring Cloud等微服务框架无缝集成,实现自动化的服务发现和负载均衡。
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,出现以下界面代表启动成功
- 然后idea中启动以下三个模块
浏览器地址栏输入:http://127.0.0.1/consumer/getUserById/520,出现以下界面代表配置成功
经过上述配置仍存在些许问题,比如拼接url和参数显得好傻,所以接下来要使用feign 处理参数拼接问题
1.3. 什么是feign
Feign是一个轻量级的HTTP客户端,它使用了基于注解的方式来定义和描述RESTful服务的接口,使得客户端代码变得更加简洁和易于维护。它的特点包括:
- 与Spring Cloud集成:Feign可以与Spring Cloud无缝集成,使得服务之间的通信更加简单和可靠。它可以自动发现服务并进行负载均衡,同时也支持服务降级和熔断等机制,保证了服务的高可用性和可靠性。
- 声明式的API:Feign使用基于注解的方式来定义和描述RESTful服务的接口,它提供了一种声明式的API,使得代码更加易于理解和维护。通过简单的注解,可以定义接口的路径、请求方式、请求参数和返回类型等信息,而无需关注具体的HTTP请求和响应细节。
- 易于扩展:Feign提供了许多自定义的扩展点,使得用户可以根据自己的需求来扩展和定制Feign的功能。例如,可以自定义编码器和解码器、请求拦截器、日志记录器等,从而满足各种不同的业务场景。
- 支持多种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,出现以下界面代表配置成功