微服务架构中,我们比较关心的就是服务间的服务改如何调用,有哪些调用方式

image.png

在springcloud中服务间调用方式主要是使用 http restful方式进行服务间调用

1、创建示例demo

1.1 创建 springcloud-consul-users-6104

1.1.1 pom.xml文件

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  3. xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
  4. <modelVersion>4.0.0</modelVersion>
  5. <parent>
  6. <groupId>org.springframework.boot</groupId>
  7. <artifactId>spring-boot-starter-parent</artifactId>
  8. <version>2.3.5.RELEASE</version>
  9. <relativePath/> <!-- lookup parent from repository -->
  10. </parent>
  11. <groupId>com.morrow</groupId>
  12. <artifactId>springcloud-consul-users-6104</artifactId>
  13. <version>0.0.1-SNAPSHOT</version>
  14. <name>springcloud-consul-users-6104</name>
  15. <description>springcloud-consul-users-6104 project for Spring Boot</description>
  16. <properties>
  17. <java.version>11</java.version>
  18. <spring-cloud.version>Hoxton.SR6</spring-cloud.version>
  19. </properties>
  20. <dependencies>
  21. <dependency>
  22. <groupId>org.springframework.boot</groupId>
  23. <artifactId>spring-boot-starter-web</artifactId>
  24. </dependency>
  25. <dependency>
  26. <groupId>org.springframework.boot</groupId>
  27. <artifactId>spring-boot-starter-test</artifactId>
  28. <scope>test</scope>
  29. </dependency>
  30. <!--引入consul依赖-->
  31. <dependency>
  32. <groupId>org.springframework.cloud</groupId>
  33. <artifactId>spring-cloud-starter-consul-discovery</artifactId>
  34. </dependency>
  35. <!-- 健康度监控-->
  36. <dependency>
  37. <groupId>org.springframework.boot</groupId>
  38. <artifactId>spring-boot-starter-actuator</artifactId>
  39. </dependency>
  40. <dependency>
  41. <groupId>org.projectlombok</groupId>
  42. <artifactId>lombok</artifactId>
  43. </dependency>
  44. </dependencies>
  45. <!--全局管理springcloud版本,并不会引入具体依赖-->
  46. <dependencyManagement>
  47. <dependencies>
  48. <dependency>
  49. <groupId>org.springframework.cloud</groupId>
  50. <artifactId>spring-cloud-dependencies</artifactId>
  51. <version>${spring-cloud.version}</version>
  52. <type>pom</type>
  53. <scope>import</scope>
  54. </dependency>
  55. </dependencies>
  56. </dependencyManagement>
  57. <build>
  58. <plugins>
  59. <plugin>
  60. <groupId>org.springframework.boot</groupId>
  61. <artifactId>spring-boot-maven-plugin</artifactId>
  62. </plugin>
  63. </plugins>
  64. </build>
  65. </project>

1.1.2 yml 文件

  1. server:
  2. port: 6104
  3. spring:
  4. application:
  5. name: users
  6. cloud:
  7. consul:
  8. host: localhost #注册consul服务的主机
  9. port: 8500 #注册consul服务的端口号
  10. discovery:
  11. # register-health-check: true # 开启 关闭consul 服务的健康检查[不推荐]
  12. service-name: ${spring.application.name} #指定注册的服务名称 默认就是应用名

1.2 创建 springcloud-consul-products-6105

1.2.1 pom.xml文件

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  3. xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
  4. <modelVersion>4.0.0</modelVersion>
  5. <parent>
  6. <groupId>org.springframework.boot</groupId>
  7. <artifactId>spring-boot-starter-parent</artifactId>
  8. <version>2.3.5.RELEASE</version>
  9. <relativePath/> <!-- lookup parent from repository -->
  10. </parent>
  11. <groupId>com.morrow</groupId>
  12. <artifactId>springcloud-consul-products-6105</artifactId>
  13. <version>0.0.1-SNAPSHOT</version>
  14. <name>springcloud-consul-products-6105</name>
  15. <description>springcloud-consul-products-6105 project for Spring Boot</description>
  16. <properties>
  17. <java.version>11</java.version>
  18. <spring-cloud.version>Hoxton.SR6</spring-cloud.version>
  19. </properties>
  20. <dependencies>
  21. <dependency>
  22. <groupId>org.springframework.boot</groupId>
  23. <artifactId>spring-boot-starter-web</artifactId>
  24. </dependency>
  25. <dependency>
  26. <groupId>org.springframework.boot</groupId>
  27. <artifactId>spring-boot-starter-test</artifactId>
  28. <scope>test</scope>
  29. </dependency>
  30. <!--引入consul依赖-->
  31. <dependency>
  32. <groupId>org.springframework.cloud</groupId>
  33. <artifactId>spring-cloud-starter-consul-discovery</artifactId>
  34. </dependency>
  35. <!-- 健康度监控-->
  36. <dependency>
  37. <groupId>org.springframework.boot</groupId>
  38. <artifactId>spring-boot-starter-actuator</artifactId>
  39. </dependency>
  40. <dependency>
  41. <groupId>org.projectlombok</groupId>
  42. <artifactId>lombok</artifactId>
  43. </dependency>
  44. </dependencies>
  45. <!--全局管理springcloud版本,并不会引入具体依赖-->
  46. <dependencyManagement>
  47. <dependencies>
  48. <dependency>
  49. <groupId>org.springframework.cloud</groupId>
  50. <artifactId>spring-cloud-dependencies</artifactId>
  51. <version>${spring-cloud.version}</version>
  52. <type>pom</type>
  53. <scope>import</scope>
  54. </dependency>
  55. </dependencies>
  56. </dependencyManagement>
  57. <build>
  58. <plugins>
  59. <plugin>
  60. <groupId>org.springframework.boot</groupId>
  61. <artifactId>spring-boot-maven-plugin</artifactId>
  62. </plugin>
  63. </plugins>
  64. </build>
  65. </project>

1.2.2 yml 文件

  1. server:
  2. port: 6105
  3. spring:
  4. application:
  5. name: products
  6. cloud:
  7. consul:
  8. host: localhost #注册consul服务的主机
  9. port: 8500 #注册consul服务的端口号
  10. discovery:
  11. # register-health-check: true # 开启 关闭consul 服务的健康检查[不推荐]
  12. service-name: ${spring.application.name} #指定注册的服务名称 默认就是应用名

1.2.3 ProductController文件

  1. @RestController
  2. @Slf4j
  3. public class ProductController {
  4. @Value("${server.port}")
  5. private int port;
  6. @GetMapping("/product/findAll")
  7. public Map<String,Object> findAll(){
  8. log.info("商品服务查询所有调用成功,当前服务端口:[{}]",port);
  9. Map<String, Object> map = new HashMap<String,Object>();
  10. map.put("msg","服务调用成功,服务提供端口为: "+port);
  11. map.put("status",true);
  12. return map;
  13. }
  14. }

2、服务间通信方式

2.1 基于 RestTemplate 的服务调用

spring框架提供的RestTemplate类可用于在应用中调用rest服务,它简化了与http服务的通信方式,统一了RESTful的标准,封装了http链接, 我们只需要传入url及返回值类型即可。相较于之前常用的HttpClient,RestTemplate是一种更优雅的调用RESTful服务的方式。

RestTemplate直接调用存在问题

1.直接使用restTemplate方式调用没有经过服务注册中心获取服务地址,代码写死不利于维护,当服务宕机时不能高效剔除

2.调用服务时没有负载均衡需要自己实现负载均衡策略

2.1.1 UserController文件

  1. @RestController
  2. @Slf4j
  3. public class UserController {
  4. @GetMapping("/user/findAll")
  5. public String findAll(){
  6. log.info("调用用户服务...");
  7. //1.使用restTemplate调用商品服务
  8. RestTemplate restTemplate = new RestTemplate();
  9. String forObject = restTemplate.getForObject("http://localhost:6105/product/findAll",
  10. String.class);
  11. return forObject;
  12. }
  13. }

image.png
image.png
image.png
image.png

2.1 基于 Ribbon 的服务调用

官方网址: https://github.com/Netflix/ribbon Spring Cloud Ribbon是一个基于HTTP和TCP的客户端负载均衡工具,它基于Netflix Ribbon实现。通过Spring Cloud的封装,可以让我们轻松地将面向服务的REST模版请求自动转换成客户端负载均衡的服务调用。

1.如果使用的是eureka client 和 consul client,无须引入依赖,因为在eureka,consul中默认集成了ribbon组件
2.如果使用的client中没有ribbon依赖需要显式引入如下依赖

org.springframework.cloud spring-cloud-starter-netflix-ribbon

image.png

2.1.1 UserController文件 使用 discovery client 进行客户端调用

  1. @Autowired
  2. private DiscoveryClient discoveryClient;
  3. @GetMapping("/user/findAll2")
  4. public String findAll2(){
  5. log.info("调用用户服务...");
  6. // 1. 使用discovery client 进行客户端调用
  7. List<ServiceInstance> serviceInstances = discoveryClient.getInstances("products");
  8. for (ServiceInstance product : serviceInstances) {
  9. log.info("服务主机:[{}]",product.getHost());
  10. log.info("服务端口:[{}]",product.getPort());
  11. log.info("服务地址:[{}]",product.getUri());
  12. log.info("====================================");
  13. }
  14. return null;
  15. }

2.1.2 UserController文件 使用 loadBalanceClient 进行客户端调用

  1. @Autowired
  2. private LoadBalancerClient loadBalancerClient;
  3. @GetMapping("/user/findAll3")
  4. public ServiceInstance findAll3(){
  5. log.info("调用用户服务...");
  6. // 2. 使用loadBalanceClient 进行客户端调用
  7. ServiceInstance serviceInstance = loadBalancerClient.choose("products");
  8. log.info("服务主机:[{}]",serviceInstance.getHost());
  9. log.info("服务端口:[{}]",serviceInstance.getPort());
  10. log.info("服务地址:[{}]",serviceInstance.getUri());
  11. return serviceInstance;
  12. }

2.1.3 UserController文件 使用 @loadBalanced 进行客户端调用

  1. @Autowired
  2. private RestTemplate restTemplate;
  3. @GetMapping("/user/findAll4")
  4. public Map findAll4(){
  5. log.info("调用用户服务...");
  6. // 3. 使用@loadBalanced 进行客户端调用
  7. // 创建配置文件 RestTemplateConfig
  8. Map object = restTemplate.getForObject("http://products/product/findAll", Map.class);
  9. return object;
  10. }

RestTemplateConfig 配置文件

  1. @Configuration
  2. public class RestTemplateConfig {
  3. @Bean
  4. @LoadBalanced
  5. public RestTemplate getRestTemplate() {
  6. return new RestTemplate();
  7. }
  8. }

2.1.4 Ribbon负载均衡策略

负载均衡算法

  • RoundRobinRule 轮训策略 按顺序循环选择 Server
  • RandomRule 随机策略 随机选择 Server
  • AvailabilityFilteringRule 可用过滤策略
    `会先过滤由于多次访问故障而处于断路器跳闸状态的服务,还有并发的连接数量超过阈值的服务,然后对剩余的服务列表按照轮询策略进行 访问
  • WeightedResponseTimeRule 响应时间加权策略
    `根据平均响应的时间计算所有服务的权重,响应时间越快服务权重越大被选中的概率越高,刚启动时如果统计信息不足,则使用
    RoundRobinRule策略,等统计信息足够会切换到
  • RetryRule 重试策略
    `先按照RoundRobinRule的策略获取服务,如果获取失败则在制定时间内进行重试,获取可用的服务。
  • BestAviableRule 最低并发策略
    `会先过滤掉由于多次访问故障而处于断路器跳闸状态的服务,然后选择一个并发量最小的服务

image.png

product 启动 6106

相关配置

-Dserver.port=6106

image.png

同时 启动 6105 6106
image.png
image.png

2.1.4.1 配置 轮训策略

yml 文件

  1. server:
  2. port: 6104
  3. spring:
  4. application:
  5. name: users
  6. cloud:
  7. consul:
  8. host: localhost #注册consul服务的主机
  9. port: 8500 #注册consul服务的端口号
  10. discovery:
  11. # register-health-check: true # 开启 关闭consul 服务的健康检查[不推荐]
  12. service-name: ${spring.application.name} #指定注册的服务名称 默认就是应用名
  13. products:
  14. ribbon:
  15. NFLoadBalancerRuleClassName: com.netflix.loadbalancer.RandomRule

Video_2021-07-19_140708.gif

其它策略.. 类似

2.1.5 Ribbon停止维护

image.png

3、示例代码

springcloud-consul-users-6104 springcloud-consul-products-6105