服务拆分和远程调用

任何分布式架构都离不开服务的拆分,微服务也是一样。

微服务拆分时的几个原则:

  • 不同微服务,不要重复开发相同业务
  • 微服务数据独立,不要访问其它微服务的数据库
  • 微服务可以将自己的业务暴露为接口,供其它微服务调用

image-20210713210800950.png

demo业务和项目架构

我们的业务需求是订单服务,和用户服务。
我们创建两个模块,一个订单模块,一个用户模块。
每个模块都是那些东西,controller,service,mapper,do,差不多的。
两个模块都有启动类,也就是有两个application.yaml配置文件。
项目的架构大概是这样的:
父maven项目包裹两个子maven项目
image.png
两个配置文件中都连接了数据库,但是不是一个数据库。两个配置文件还运行在不同的端口上面,模拟分布式

  1. server:
  2. port: 8080
  3. spring:
  4. datasource:
  5. url: jdbc:mysql://localhost:3306/cloud_order?useSSL=false
  6. username: root
  7. password: 123456
  8. driver-class-name: com.mysql.jdbc.Driver
  1. server:
  2. port: 8081
  3. spring:
  4. datasource:
  5. url: jdbc:mysql://localhost:3306/cloud_user?useSSL=false
  6. username: root
  7. password: 123456
  8. driver-class-name: com.mysql.jdbc.Driver

SQL文件

  1. create database cloud_order
  2. use cloud_order
  3. SET NAMES utf8mb4;
  4. SET FOREIGN_KEY_CHECKS = 0;
  5. -- ----------------------------
  6. -- Table structure for tb_order
  7. -- ----------------------------
  8. DROP TABLE IF EXISTS `tb_order`;
  9. CREATE TABLE `tb_order` (
  10. `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT '订单id',
  11. `user_id` bigint(20) NOT NULL COMMENT '用户id',
  12. `name` varchar(100) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL COMMENT '商品名称',
  13. `price` bigint(20) NOT NULL COMMENT '商品价格',
  14. `num` int(10) NULL DEFAULT 0 COMMENT '商品数量',
  15. PRIMARY KEY (`id`) USING BTREE,
  16. UNIQUE INDEX `username`(`name`) USING BTREE
  17. ) ENGINE = InnoDB AUTO_INCREMENT = 109 CHARACTER SET = utf8 COLLATE = utf8_general_ci ROW_FORMAT = Compact;
  18. -- ----------------------------
  19. -- Records of tb_order
  20. -- ----------------------------
  21. INSERT INTO `tb_order` VALUES (101, 1, 'Apple 苹果 iPhone 12 ', 699900, 1);
  22. INSERT INTO `tb_order` VALUES (102, 2, '雅迪 yadea 新国标电动车', 209900, 1);
  23. INSERT INTO `tb_order` VALUES (103, 3, '骆驼(CAMEL)休闲运动鞋女', 43900, 1);
  24. INSERT INTO `tb_order` VALUES (104, 4, '小米10 双模5G 骁龙865', 359900, 1);
  25. INSERT INTO `tb_order` VALUES (105, 5, 'OPPO Reno3 Pro 双模5G 视频双防抖', 299900, 1);
  26. INSERT INTO `tb_order` VALUES (106, 6, '美的(Midea) 新能效 冷静星II ', 544900, 1);
  27. INSERT INTO `tb_order` VALUES (107, 2, '西昊/SIHOO 人体工学电脑椅子', 79900, 1);
  28. INSERT INTO `tb_order` VALUES (108, 3, '梵班(FAMDBANN)休闲男鞋', 31900, 1);
  29. SET FOREIGN_KEY_CHECKS = 1;
  1. create database cloud_user
  2. use cloud_user
  3. SET NAMES utf8mb4;
  4. SET FOREIGN_KEY_CHECKS = 0;
  5. -- ----------------------------
  6. -- Table structure for tb_user
  7. -- ----------------------------
  8. DROP TABLE IF EXISTS `tb_user`;
  9. CREATE TABLE `tb_user` (
  10. `id` bigint(20) NOT NULL AUTO_INCREMENT,
  11. `username` varchar(100) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL COMMENT '收件人',
  12. `address` varchar(255) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL COMMENT '地址',
  13. PRIMARY KEY (`id`) USING BTREE,
  14. UNIQUE INDEX `username`(`username`) USING BTREE
  15. ) ENGINE = InnoDB AUTO_INCREMENT = 109 CHARACTER SET = utf8 COLLATE = utf8_general_ci ROW_FORMAT = Compact;
  16. -- ----------------------------
  17. -- Records of tb_user
  18. -- ----------------------------
  19. INSERT INTO `tb_user` VALUES (1, '柳岩', '湖南省衡阳市');
  20. INSERT INTO `tb_user` VALUES (2, '文二狗', '陕西省西安市');
  21. INSERT INTO `tb_user` VALUES (3, '华沉鱼', '湖北省十堰市');
  22. INSERT INTO `tb_user` VALUES (4, '张必沉', '天津市');
  23. INSERT INTO `tb_user` VALUES (5, '郑爽爽', '辽宁省沈阳市大东区');
  24. INSERT INTO `tb_user` VALUES (6, '范兵兵', '山东省青岛市');
  25. SET FOREIGN_KEY_CHECKS = 1;

pom.xml

父工程 pom.xml

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <project xmlns="http://maven.apache.org/POM/4.0.0"
  3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  4. xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
  5. <modelVersion>4.0.0</modelVersion>
  6. <groupId>cn.itcast.demo</groupId>
  7. <artifactId>cloud-demo</artifactId>
  8. <version>1.0</version>
  9. <modules>
  10. <module>user-service</module>
  11. <module>order-service</module>
  12. </modules>
  13. <packaging>pom</packaging>
  14. <parent>
  15. <groupId>org.springframework.boot</groupId>
  16. <artifactId>spring-boot-starter-parent</artifactId>
  17. <version>2.3.9.RELEASE</version>
  18. <relativePath/>
  19. </parent>
  20. <properties>
  21. <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
  22. <project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding>
  23. <java.version>1.8</java.version>
  24. <spring-cloud.version>Hoxton.SR10</spring-cloud.version>
  25. <mysql.version>5.1.47</mysql.version>
  26. <mybatis.version>2.1.1</mybatis.version>
  27. </properties>
  28. <dependencyManagement>
  29. <dependencies>
  30. <!-- springCloud -->
  31. <dependency>
  32. <groupId>org.springframework.cloud</groupId>
  33. <artifactId>spring-cloud-dependencies</artifactId>
  34. <version>${spring-cloud.version}</version>
  35. <type>pom</type>
  36. <scope>import</scope>
  37. </dependency>
  38. <!-- mysql驱动 -->
  39. <dependency>
  40. <groupId>mysql</groupId>
  41. <artifactId>mysql-connector-java</artifactId>
  42. <version>${mysql.version}</version>
  43. </dependency>
  44. <!--mybatis-->
  45. <dependency>
  46. <groupId>org.mybatis.spring.boot</groupId>
  47. <artifactId>mybatis-spring-boot-starter</artifactId>
  48. <version>${mybatis.version}</version>
  49. </dependency>
  50. </dependencies>
  51. </dependencyManagement>
  52. <dependencies>
  53. <dependency>
  54. <groupId>org.projectlombok</groupId>
  55. <artifactId>lombok</artifactId>
  56. </dependency>
  57. </dependencies>
  58. </project>

order-service pom.xml

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <project xmlns="http://maven.apache.org/POM/4.0.0"
  3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  4. xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
  5. <parent>
  6. <artifactId>cloud-demo</artifactId>
  7. <groupId>cn.itcast.demo</groupId>
  8. <version>1.0</version>
  9. </parent>
  10. <modelVersion>4.0.0</modelVersion>
  11. <artifactId>order-service</artifactId>
  12. <dependencies>
  13. <dependency>
  14. <groupId>org.springframework.boot</groupId>
  15. <artifactId>spring-boot-starter-web</artifactId>
  16. </dependency>
  17. <dependency>
  18. <groupId>mysql</groupId>
  19. <artifactId>mysql-connector-java</artifactId>
  20. </dependency>
  21. <!--mybatis-->
  22. <dependency>
  23. <groupId>org.mybatis.spring.boot</groupId>
  24. <artifactId>mybatis-spring-boot-starter</artifactId>
  25. </dependency>
  26. </dependencies>
  27. <build>
  28. <plugins>
  29. <plugin>
  30. <groupId>org.springframework.boot</groupId>
  31. <artifactId>spring-boot-maven-plugin</artifactId>
  32. </plugin>
  33. </plugins>
  34. </build>
  35. </project>

user-service pom.xml

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <project xmlns="http://maven.apache.org/POM/4.0.0"
  3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  4. xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
  5. <parent>
  6. <artifactId>cloud-demo</artifactId>
  7. <groupId>cn.itcast.demo</groupId>
  8. <version>1.0</version>
  9. </parent>
  10. <modelVersion>4.0.0</modelVersion>
  11. <artifactId>user-service</artifactId>
  12. <dependencies>
  13. <dependency>
  14. <groupId>org.springframework.boot</groupId>
  15. <artifactId>spring-boot-starter-web</artifactId>
  16. </dependency>
  17. <dependency>
  18. <groupId>mysql</groupId>
  19. <artifactId>mysql-connector-java</artifactId>
  20. </dependency>
  21. <!--mybatis-->
  22. <dependency>
  23. <groupId>org.mybatis.spring.boot</groupId>
  24. <artifactId>mybatis-spring-boot-starter</artifactId>
  25. </dependency>
  26. </dependencies>
  27. <build>
  28. <finalName>app</finalName>
  29. <plugins>
  30. <plugin>
  31. <groupId>org.springframework.boot</groupId>
  32. <artifactId>spring-boot-maven-plugin</artifactId>
  33. </plugin>
  34. </plugins>
  35. </build>
  36. </project>

接口

订单服务的接口

RESTful风格的接口,根据订单id返回订单信息。

  1. package cn.itcast.order.web;
  2. import cn.itcast.order.pojo.Order;
  3. import cn.itcast.order.service.OrderService;
  4. import org.springframework.beans.factory.annotation.Autowired;
  5. import org.springframework.web.bind.annotation.GetMapping;
  6. import org.springframework.web.bind.annotation.PathVariable;
  7. import org.springframework.web.bind.annotation.RequestMapping;
  8. import org.springframework.web.bind.annotation.RestController;
  9. @RestController
  10. @RequestMapping("order")
  11. public class OrderController {
  12. @Autowired
  13. private OrderService orderService;
  14. @GetMapping("{orderId}")
  15. public Order queryOrderByUserId(@PathVariable("orderId") Long orderId) {
  16. // 根据id查询订单并返回
  17. return orderService.queryOrderById(orderId);
  18. }
  19. }

image.png

用户服务的接口

根据用户id查询用户信息

  1. package cn.itcast.user.web;
  2. import cn.itcast.user.pojo.User;
  3. import cn.itcast.user.service.UserService;
  4. import lombok.extern.slf4j.Slf4j;
  5. import org.springframework.beans.factory.annotation.Autowired;
  6. import org.springframework.web.bind.annotation.*;
  7. @Slf4j
  8. @RestController
  9. @RequestMapping("/user")
  10. public class UserController {
  11. @Autowired
  12. private UserService userService;
  13. /**
  14. * 路径: /user/110
  15. *
  16. * @param id 用户id
  17. * @return 用户
  18. */
  19. @GetMapping("/{id}")
  20. public User queryById(@PathVariable("id") Long id) {
  21. return userService.queryById(id);
  22. }
  23. }

image.png

订单需要显示用户信息怎么办?

传统的做法是在service层通过orderMapper去查出订单信息,根据查出来的订单信息里面的用户id,再去调userMapper查询出用户信息,最后组成vo返回。

这样的话,在订单服务里面就做了用户服务应该做的事,违背了微服务的初衷。记住单一职责,订单服务就对订单进行操作,用户服务就对用户进行操作。

项目和数据库都隔离了。指定是用不了传统的方法。
我们查询完订单以后通过网络请求去访问查询用户的接口,就获取到了用户信息,然后和订单信息组装起来。
这就是最基础的服务之间的调用,用的http协议,爬虫就是这样的。
image-20210713213312278.png

实现远程调用

注册RestTemplate,需要在配置类里面注册

  1. package cn.itcast.order.config;
  2. import org.springframework.context.annotation.Bean;
  3. import org.springframework.context.annotation.Configuration;
  4. import org.springframework.web.client.RestTemplate;
  5. @Configuration
  6. public class MyConfig {
  7. @Bean
  8. public RestTemplate restTemplate() {
  9. return new RestTemplate();
  10. }
  11. }

修改订单的service层

  1. package cn.itcast.order.service;
  2. import cn.itcast.order.mapper.OrderMapper;
  3. import cn.itcast.order.pojo.Order;
  4. import cn.itcast.order.pojo.User;
  5. import org.springframework.beans.factory.annotation.Autowired;
  6. import org.springframework.stereotype.Service;
  7. import org.springframework.web.client.RestTemplate;
  8. @Service
  9. public class OrderService {
  10. @Autowired
  11. private OrderMapper orderMapper;
  12. @Autowired
  13. private RestTemplate restTemplate;
  14. public Order queryOrderById(Long orderId) {
  15. // 1.查询订单
  16. Order order = orderMapper.findById(orderId);
  17. // 2.远程查询user
  18. String url = "http://localhost:8081/user/"+order.getUserId();
  19. User user = restTemplate.getForObject(url, User.class);
  20. // 3.组合
  21. order.setUser(user);
  22. // 4.返回
  23. return order;
  24. }
  25. }

image.png

提供者与消费者

在服务调用关系中,会有两个不同的角色:
服务提供者:一次业务中,被其它微服务调用的服务。(提供接口给其它微服务)
服务消费者:一次业务中,调用其它微服务的服务。(调用其它微服务提供的接口)
image-20210713214404481.png
但是,服务提供者与服务消费者的角色并不是绝对的,而是相对于业务而言。
如果服务A调用了服务B,而服务B又调用了服务C,服务B的角色是什么?

  • 对于A调用B的业务而言:A是服务消费者,B是服务提供者
  • 对于B调用C的业务而言:B是服务消费者,C是服务提供者

因此,服务B既可以是服务提供者,也可以是服务消费者。