微服务拆分

张裕  金牌会员 | 2025-1-17 16:30:54 | 显示全部楼层 | 阅读模式
打印 上一主题 下一主题

主题 566|帖子 566|积分 1698

微服务拆分

接下来,我们就一起将黑马商城这个单体项目拆分为微服务项目,并解决此中出现的各种问题。
熟悉黑马商城

首先,我们需要熟悉黑马商城项目的基本布局:

大家可以直接启动该项目,测试效果。不过,需要修改数据库毗连参数,在application-local.yaml中:
  1. hm:
  2.   db:
  3.     host: 192.168.202.128 # 修改为你自己的虚拟机IP地址
  4.     pw: 1234 # 修改为docker中的MySQL密码
复制代码
同时设置启动项激活的是local环境:

登录

首先来看一下登录业务流程:

登录入口在com.hmall.controller.UserController中的login方法:

搜索商品

在首页搜索框输入关键字,点击搜索即可进入搜索列表页面:

该页面会调用接口:/search/list,对应的服务端入口在com.hmall.controller.SearchController中的search方法:

这里如今是使用数据库实现了简单的分页查询。
购物车

在搜索到的商品列表中,点击按钮参加购物车,即可将商品参加购物车:

参加乐成后即可进入购物车列表页,查察本身购物车商品列表:

同时这里还可以对购物车实现修改、删除等操纵。
干系功能全部在com.hmall.controller.CartController中:

此中,查询购物车列表时,由于要判定商品最新的价格和状态,所以还需要查询商品信息,业务流程如下:

下单

在购物车页面点击结算按钮,会进入订单结算页面:

点击提交订单,会提交哀求到服务端,服务端做3件事变:


  • 创建一个新的订单
  • 扣减商品库存
  • 清算购物车中商品
业务入口在com.hmall.controller.OrderController中的createOrder方法:

付出

下单完成后会跳转到付出页面,如今只支持余额付出

在选择余额付出这种方式后,会发起哀求到服务端,服务端会立刻创建一个付出流水单,并返回付出流水单号到前端。
当用户输入用户暗码,然后点击确认付出时,页面会发送哀求到服务端,而服务端会做几件事变:


  • 校验用户暗码
  • 扣减余额
  • 修改付出流水状态
  • 修改交易订单状态
哀求入口在com.hmall.controller.PayController中:

服务拆分原则

服务拆分一定要考虑几个问题:


  • 什么时间拆?
  • 如何拆?
什么时间拆

一样平常情况下,对于一个初创的项目,首先要做的是验证项目的可行性。因此这一阶段的首要任务是敏捷开辟,快速产出生产可用的产物,投入市场做验证。为了达成这一目的,该阶段项目架构通常会比力简单,许多情况下会直接采用单体架构,如许开辟成本比力低,可以快速产出结果,一旦发现项目不符合市场,损失较小。如果这一阶段采用复杂的微服务架构,投入大量的人力和时间成本用于架构计划,终极发现产物不符合市场需求,等于全部做了无勤奋。
所以,对于大多数小型项目来说,一样平常是先采用单体架构,随着用户规模扩大、业务复杂后再逐渐拆分为微服务架构**。如许初期成本会比力低,可以快速试错。但是,这么做的问题就在于后期做服务拆分时,可能会碰到许多代码耦合带来的问题,拆分比力困难(**前易后难)。
而对于一些大型项目,在立项之初目的就很明白,为了长远考虑,在架构计划时就直接选择微服务架构。固然前期投入较多,但后期就少了拆分服务的烦恼(前难后易)。
怎么拆

之前我们说过,微服务拆分时粒度要小,这实在是拆分的目标。具体可以从两个角度来分析:


  • 高内聚:每个微服务的职责要只管单一,包含的业务相互关联度高、完整度高。
  • 低耦合:每个微服务的功能要相对独立,只管减少对其它微服务的依赖,大概依赖接口的稳定性要强。
高内聚首先是单一职责,但不能说一个微服务就一个接口,而是要包管微服务内部业务的完整性为前提。目标是当我们要修改某个业务时,最好就只修改当前微服务,如许变更的成本更低。一旦微服务做到了高内聚,那么服务之间的耦合度自然就降低了。
当然,微服务之间不可避免的会有或多或少的业务交互,比如下单时需要查询商品数据。这个时间我们不能在订单服务直接查询商品数据库,否则就导致了数据耦合。而应该由商品服务对应暴露接口,并且一定要包管微服务对外接口的稳定性(即:只管包管接口外观稳定)。固然出现了服务间调用,但此时无论你如何在商品服务做内部修改,都不会影响到订单微服务,服务间的耦合度就降低了。
明白了拆分目标,接下来就是拆分方式了。我们在做服务拆分时一样平常有两种方式:


  • 纵向拆分
  • 横向拆分
所谓纵向拆分,就是按照项目的功能模块来拆分。比方黑马商城中,就有用户管理功能、订单管理功能、购物车功能、商品管理功能、付出功能等。那么按照功能模块将他们拆分为一个个服务,就属于纵向拆分。这种拆分模式可以尽可能提高服务的内聚性。
横向拆分,是看各个功能模块之间有没有公共的业务部分,如果有将其抽取出来作为通用服务。比方用户登录是需要发送消息通知,纪录风控数据,下单时也要发送短信,纪录风控数据。因此消息发送、风控数据纪录就是通用的业务功能,因此可以将他们分别抽取为公共服务:消息中央服务、风控管理服务。如许可以提高业务的复用性,避免重复开辟。同时通用业务一样平常接口稳定性较强,也不会使服务之间太过耦合。
当然,由于黑马商城并不是一个完整的项目,此中的短信发送、风控管理并没有实现,这里就不再考虑了。而其它的业务按照纵向拆分,可以分为以下几个微服务:


  • 用户服务
  • 商品服务
  • 订单服务
  • 购物车服务
  • 付出服务
拆分购物车、商品服务

接下来,我们先把商品管理功能、购物车功能抽取为两个独立服务。
一样平常微服务项目有两种差别的工程布局:


  • 完全解耦:每一个微服务都创建为一个独立的工程,乃至可以使用差别的开辟语言来开辟,项目完全解耦。

    • 长处:服务之间耦合度低
    • 缺点:每个项目都有本身的独立堆栈,管理起来比力麻烦

  • Maven聚合:整个项目为一个Project,然后每个微服务是此中的一个Module

    • 长处:项目代码会合,管理和运维方便
    • 缺点:服务之间耦合,编译时间较长

注意
这里采用Maven聚合工程,大家以后到了企业,可以根据需求自由选择工程布局。在hmall父工程之中,我已经提前界说了SpringBoot、SpringCloud的依赖版本,所以为了方便期间,我们直接在这个项目中创建微服务module.
商品服务

在hmall中创建module,选择maven模块,并设定JDK版本为11。商品模块,我们起名为item-service:

引入依赖:
  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>hmall</artifactId>
  7.         <groupId>com.heima</groupId>
  8.         <version>1.0.0</version>
  9.     </parent>
  10.     <modelVersion>4.0.0</modelVersion>
  11.     <artifactId>item-service</artifactId>
  12.     <properties>
  13.         <maven.compiler.source>11</maven.compiler.source>
  14.         <maven.compiler.target>11</maven.compiler.target>
  15.     </properties>
  16.     <dependencies>
  17.         <!--common-->
  18.         <dependency>
  19.             <groupId>com.heima</groupId>
  20.             <artifactId>hm-common</artifactId>
  21.             <version>1.0.0</version>
  22.         </dependency>
  23.         <!--web-->
  24.         <dependency>
  25.             <groupId>org.springframework.boot</groupId>
  26.             <artifactId>spring-boot-starter-web</artifactId>
  27.         </dependency>
  28.         <!--数据库-->
  29.         <dependency>
  30.             <groupId>mysql</groupId>
  31.             <artifactId>mysql-connector-java</artifactId>
  32.         </dependency>
  33.         <!--mybatis-->
  34.         <dependency>
  35.             <groupId>com.baomidou</groupId>
  36.             <artifactId>mybatis-plus-boot-starter</artifactId>
  37.         </dependency>
  38.         <!--单元测试-->
  39.         <dependency>
  40.             <groupId>org.springframework.boot</groupId>
  41.             <artifactId>spring-boot-starter-test</artifactId>
  42.         </dependency>
  43.     </dependencies>
  44.     <build>
  45.         <finalName>${project.artifactId}</finalName>
  46.         <plugins>
  47.             <plugin>
  48.                 <groupId>org.springframework.boot</groupId>
  49.                 <artifactId>spring-boot-maven-plugin</artifactId>
  50.             </plugin>
  51.         </plugins>
  52.     </build>
  53. </project>
复制代码
编写启动类:

代码如下:
  1. package com.hmall.item;
  2. import org.mybatis.spring.annotation.MapperScan;
  3. import org.springframework.boot.SpringApplication;
  4. import org.springframework.boot.autoconfigure.SpringBootApplication;
  5. @MapperScan("com.hmall.item.mapper")
  6. @SpringBootApplication
  7. public class ItemApplication {
  8.     public static void main(String[] args) {
  9.         SpringApplication.run(ItemApplication.class, args);
  10.     }
  11. }
复制代码
接下来是设置文件,可以从hm-service中拷贝:

此中,application.yaml内容如下:
  1. server:
  2.   #  port: 8080
  3.   port: 8081
  4. spring:
  5.   application:
  6.     name: item-service
  7.     # name: hm-service
  8.   profiles:
  9.     active: dev
  10.   datasource:
  11.     url: jdbc:mysql://${hm.db.host}:3306/hmall?useUnicode=true&characterEncoding=UTF-8&autoReconnect=true&serverTimezone=Asia/Shanghai
  12.     driver-class-name: com.mysql.cj.jdbc.Driver
  13.     username: root
  14.     password: ${hm.db.pw}
  15. mybatis-plus:
  16.   configuration:
  17.     default-enum-type-handler: com.baomidou.mybatisplus.core.handlers.MybatisEnumTypeHandler
  18.   global-config:
  19.     db-config:
  20.       update-strategy: not_null
  21.       id-type: auto
  22. logging:
  23.   level:
  24.     com.hmall: debug
  25.   pattern:
  26.     dateformat: HH:mm:ss:SSS
  27.   file:
  28.     path: "logs/${spring.application.name}"
  29. knife4j:
  30.   enable: true
  31.   openapi:
  32.     title: 商品服务接口
  33.     # title: 黑马商城接口文档
  34.     description: "信息"
  35.     #description: "黑马商城接口文档"
  36.     email: zs0536@stu.edu.cn
  37.     concat: Charlie
  38.     url: https://www.itcast.cn
  39.     version: v1.0.0
  40.     group:
  41.       default:
  42.         group-name: default
  43.         api-rule: package
  44.         api-rule-resources:
  45.           - com.hmall.item.controller
  46. #          - com.hmall.controller
复制代码
剩下的application-dev.yaml和application-local.yaml直接从hm-service拷贝即可。
然后拷贝hm-service中与商品管理有关的代码到item-service,如图:

这里有一个地方的代码需要改动,就是ItemServiceImpl中的deductStock方法:
改动前

改动后

这也是因为ItemMapper的所在包发生了变化,因此这里代码必须修改包路径。
最后,还要导入数据库表。默认的数据库毗连的是假造机,在你docker数据库执行网盘中提供的SQL文件:
   通过网盘分享的文件:微服务独立数据库
链接: https://pan.baidu.com/s/1ep9hMO6m8st8nEHedW-R6w?pwd=j6am 提取码: j6am
–来自百度网盘超等会员v5的分享
  

终极,会在数据库创建一个名为hm-item的database,将来的每一个微服务都会有本身的一个database:

注意:在企业开辟的生产环境中,每一个微服务都应该有本身的独立数据库服务,而不仅仅是database,讲堂我们用database来取代。
接下来,就可以启动测试了,在启动前我们要设置一下启动项,让默认激活的设置为local而不是dev,在打开的编辑框填写active profiles:

接着,启动item-service,访问商品微服务的swagger接口文档:http://localhost:8081/doc.html
然后测试此中的根据id批量查询商品这个接口,测试参数:100002672302,100002624500,100002533430,结果如下:

阐明商品微服务抽取乐成了。
购物车服务

与商品服务类似,在hmall下创建一个新的module,起名为cart-service:

然后是依赖:
  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>hmall</artifactId>
  7.         <groupId>com.heima</groupId>
  8.         <version>1.0.0</version>
  9.     </parent>
  10.     <modelVersion>4.0.0</modelVersion>
  11.     <artifactId>cart-service</artifactId>
  12.     <properties>
  13.         <maven.compiler.source>11</maven.compiler.source>
  14.         <maven.compiler.target>11</maven.compiler.target>
  15.     </properties>
  16.     <dependencies>
  17.         <!--common-->
  18.         <dependency>
  19.             <groupId>com.heima</groupId>
  20.             <artifactId>hm-common</artifactId>
  21.             <version>1.0.0</version>
  22.         </dependency>
  23.         <!--web-->
  24.         <dependency>
  25.             <groupId>org.springframework.boot</groupId>
  26.             <artifactId>spring-boot-starter-web</artifactId>
  27.         </dependency>
  28.         <!--数据库-->
  29.         <dependency>
  30.             <groupId>mysql</groupId>
  31.             <artifactId>mysql-connector-java</artifactId>
  32.         </dependency>
  33.         <!--mybatis-->
  34.         <dependency>
  35.             <groupId>com.baomidou</groupId>
  36.             <artifactId>mybatis-plus-boot-starter</artifactId>
  37.         </dependency>
  38.         <!--单元测试-->
  39.         <dependency>
  40.             <groupId>org.springframework.boot</groupId>
  41.             <artifactId>spring-boot-starter-test</artifactId>
  42.         </dependency>
  43.     </dependencies>
  44.     <build>
  45.         <finalName>${project.artifactId}</finalName>
  46.         <plugins>
  47.             <plugin>
  48.                 <groupId>org.springframework.boot</groupId>
  49.                 <artifactId>spring-boot-maven-plugin</artifactId>
  50.             </plugin>
  51.         </plugins>
  52.     </build>
  53. </project>
复制代码
然后是启动类:
  1. package com.hmall.cart;
  2. import org.mybatis.spring.annotation.MapperScan;
  3. import org.springframework.boot.SpringApplication;
  4. import org.springframework.boot.autoconfigure.SpringBootApplication;
  5. @MapperScan("com.hmall.cart.mapper")
  6. @SpringBootApplication
  7. public class CartApplication {
  8.     public static void main(String[] args) {
  9.         SpringApplication.run(CartApplication.class, args);
  10.     }
  11. }
复制代码
然后是设置文件,同样可以拷贝自item-service,不过此中的application.yaml需要修改:
  1. server:
  2.   port: 8082
  3. spring:
  4.   application:
  5.     name: cart-service
  6.   profiles:
  7.     active: dev
  8.   datasource:
  9.     url: jdbc:mysql://${hm.db.host}:3306/hm-cart?useUnicode=true&characterEncoding=UTF-8&autoReconnect=true&serverTimezone=Asia/Shanghai
  10.     driver-class-name: com.mysql.cj.jdbc.Driver
  11.     username: root
  12.     password: ${hm.db.pw}
  13. mybatis-plus:
  14.   configuration:
  15.     default-enum-type-handler: com.baomidou.mybatisplus.core.handlers.MybatisEnumTypeHandler
  16.   global-config:
  17.     db-config:
  18.       update-strategy: not_null
  19.       id-type: auto
  20. logging:
  21.   level:
  22.     com.hmall: debug
  23.   pattern:
  24.     dateformat: HH:mm:ss:SSS
  25.   file:
  26.     path: "logs/${spring.application.name}"
  27. knife4j:
  28.   enable: true
  29.   openapi:
  30.     title: 购物车服务接口
  31.     description: "信息"
  32.     email: zs0536@stu.edu.cn
  33.     concat: Charlie
  34.     url: https://www.itcast.cn
  35.     version: v1.0.0
  36.     group:
  37.       default:
  38.         group-name: default
  39.         api-rule: package
  40.         api-rule-resources:
  41.           - com.hmall.cart.controller
复制代码
最后,把hm-service中的与购物车有关功能拷贝过来,终极的项目布局如下:

特殊注意的是com.hmall.cart.service.impl.CartServiceImpl,此中有两个地方需要处理:


  • 需要获取登录用户信息,但登录校验功能如今没有复制过来,先写死固定用户id
  • 查询购物车时需要查询商品信息,而商品信息不在当前服务,需要先将这部分代码解释

我们对这部分代码做如下修改:
  1. package com.hmall.cart.service.impl;
  2. import cn.hutool.core.util.StrUtil;
  3. import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper;
  4. import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
  5. import com.hmall.cart.domain.dto.CartFormDTO;
  6. import com.hmall.cart.domain.po.Cart;
  7. import com.hmall.cart.domain.vo.CartVO;
  8. import com.hmall.cart.mapper.CartMapper;
  9. import com.hmall.cart.service.ICartService;
  10. import com.hmall.common.exception.BizIllegalException;
  11. import com.hmall.common.utils.BeanUtils;
  12. import com.hmall.common.utils.CollUtils;
  13. import com.hmall.common.utils.UserContext;
  14. import lombok.RequiredArgsConstructor;
  15. import org.springframework.stereotype.Service;
  16. import java.util.Collection;
  17. import java.util.List;
  18. /**
  19. * <p>
  20. * 订单详情表 服务实现类
  21. * </p>
  22. *
  23. * @author 虎哥
  24. * @since 2023-05-05
  25. */
  26. @Service
  27. @RequiredArgsConstructor
  28. public class CartServiceImpl extends ServiceImpl<CartMapper, Cart> implements ICartService {
  29.     // private final IItemService itemService;
  30.     @Override
  31.     public void addItem2Cart(CartFormDTO cartFormDTO) {
  32.         // 1.获取登录用户
  33.         Long userId = UserContext.getUser();
  34.         // 2.判断是否已经存在
  35.         if (checkItemExists(cartFormDTO.getItemId(), userId)) {
  36.             // 2.1.存在,则更新数量
  37.             baseMapper.updateNum(cartFormDTO.getItemId(), userId);
  38.             return;
  39.         }
  40.         // 2.2.不存在,判断是否超过购物车数量
  41.         checkCartsFull(userId);
  42.         // 3.新增购物车条目
  43.         // 3.1.转换PO
  44.         Cart cart = BeanUtils.copyBean(cartFormDTO, Cart.class);
  45.         // 3.2.保存当前用户
  46.         cart.setUserId(userId);
  47.         // 3.3.保存到数据库
  48.         save(cart);
  49.     }
  50.     @Override
  51.     public List<CartVO> queryMyCarts() {
  52.         // 1.查询我的购物车列表
  53.         List<Cart> carts = lambdaQuery().eq(Cart::getUserId, 1L /*TODO UserContext.getUser()*/).list();
  54.         if (CollUtils.isEmpty(carts)) {
  55.             return CollUtils.emptyList();
  56.         }
  57.         // 2.转换VO
  58.         List<CartVO> vos = BeanUtils.copyList(carts, CartVO.class);
  59.         // 3.处理VO中的商品信息
  60.         handleCartItems(vos);
  61.         // 4.返回
  62.         return vos;
  63.     }
  64.     private void handleCartItems(List<CartVO> vos) {
  65.         // 1.获取商品id TODO 处理商品信息
  66.         /*Set<Long> itemIds = vos.stream().map(CartVO::getItemId).collect(Collectors.toSet());
  67.         // 2.查询商品
  68.         List<ItemDTO> items = itemService.queryItemByIds(itemIds);
  69.         if (CollUtils.isEmpty(items)) {
  70.             throw new BadRequestException("购物车中商品不存在!");
  71.         }
  72.         // 3.转为 id 到 item的map
  73.         Map<Long, ItemDTO> itemMap = items.stream().collect(Collectors.toMap(ItemDTO::getId, Function.identity()));
  74.         // 4.写入vo
  75.         for (CartVO v : vos) {
  76.             ItemDTO item = itemMap.get(v.getItemId());
  77.             if (item == null) {
  78.                 continue;
  79.             }
  80.             v.setNewPrice(item.getPrice());
  81.             v.setStatus(item.getStatus());
  82.             v.setStock(item.getStock());
  83.         }*/
  84.     }
  85.     @Override
  86.     public void removeByItemIds(Collection<Long> itemIds) {
  87.         // 1.构建删除条件,userId和itemId
  88.         QueryWrapper<Cart> queryWrapper = new QueryWrapper<Cart>();
  89.         queryWrapper.lambda()
  90.                 .eq(Cart::getUserId, UserContext.getUser())
  91.                 .in(Cart::getItemId, itemIds);
  92.         // 2.删除
  93.         remove(queryWrapper);
  94.     }
  95.     private void checkCartsFull(Long userId) {
  96.         int count = lambdaQuery().eq(Cart::getUserId, userId).count();
  97.         if (count >= 10) {
  98.             throw new BizIllegalException(StrUtil.format("用户购物车课程不能超过{}", 10));
  99.         }
  100.     }
  101.     private boolean checkItemExists(Long itemId, Long userId) {
  102.         int count = lambdaQuery()
  103.                 .eq(Cart::getUserId, userId)
  104.                 .eq(Cart::getItemId, itemId)
  105.                 .count();
  106.         return count > 0;
  107.     }
  108. }
复制代码
最后,还是要导入数据库表,在资料对应的SQL文件:

在数据库中会出现名为hm-cart的database,以及此中的cart表,代表购物车:

接下来,就可以测试了。不过在启动前,同样要设置启动项的active profile为local:

然后启动CartApplication,访问swagger文档页面:http://localhost:8082/doc.html
我们测试此中的查询我的购物车列表接口,无需填写参数,直接访问:

我们注意到,此中与商品有关的几个字段值都为空!这就是因为刚才我们解释掉了查询购物车时,查询商品信息的干系代码。
那么,我们该如何在cart-service服务中实现对item-service服务的查询呢?
服务调用

在拆分的时间,我们发现一个问题:就是购物车业务中需要查询商品信息,但商品信息查询的逻辑全部迁移到了item-service服务,导致我们无法查询。
终极结果就是查询到的购物车数据不完整,因此要想解决这个问题,我们就必须改造此中的代码,把原来本地方法调用,改造成跨微服务的远程调用(RPC,即Remote Produce Call)。
因此,如今查询购物车列表的流程变成了如许:

代码中需要变化的就是这一步:

那么问题来了:我们该如何跨服务调用,准确的说,如何在cart-service中获取item-service服务中的提供的商品数据呢?
大家思索一下,我们以前有没有实现过类似的远程查询的功能呢?
答案是肯定的,我们前端向服务端查询数据,实在就是从浏览器远程查询服务端数据。比如我们刚才通过Swagger测试商品查询接口,就是向http://localhost:8081/items这个接口发起的哀求:

而这种查询就是通过http哀求的方式来完成的,不仅仅可以实现远程查询,还可以实现新增、删除等各种远程哀求。
如果我们在cart-service中能模拟浏览器,发送http哀求到item-service,是不是就实现了跨微服务的远程调用了呢?
那么:我们该如何用Java代码发送Http的哀求呢?
RestTemplate

Spring给我们提供了一个RestTemplate的API,可以方便的实现Http哀求的发送。
   org.springframework.web.client public class RestTemplate
  extends InterceptingHttpAccessor
  implements RestOperations
  ----------------------------------------------------------------------------------------------------------------
  同步客户端执行HTTP哀求,在底层HTTP客户端库(如JDK HttpURLConnection、Apache HttpComponents等)上公开一个简单的模板方法API。RestTemplate通过HTTP方法为常见场景提供了模板,此外还提供了支持不太常见情况的通用互换和执行方法。 RestTemplate通常用作共享组件。然而,它的设置不支持并发修改,因此它的设置通常是在启动时准备的。如果需要,您可以在启动时创建多个差别设置的RestTemplate实例。如果这些实例需要共享HTTP客户端资源,它们可以使用相同的底层ClientHttpRequestFactory。 注意:从5.0开始,这个类处于维护模式,只有对更改和错误的小哀求才会被继承。请考虑使用org.springframework.web.react .client. webclient,它有更当代的API,支持同步、异步和流场景。
  ----------------------------------------------------------------------------------------------------------------
  自: 3.0 拜见: HttpMessageConverter, RequestCallback, ResponseExtractor, ResponseErrorHandler
  此中提供了大量的方法,方便我们发送Http哀求,比方:

可以看到常见的Get、Post、Put、Delete哀求都支持,如果哀求参数比力复杂,还可以使用exchange方法来构造哀求。
我们在cart-service服务中界说一个设置类:

先将RestTemplate注册为一个Bean:
  1. package com.hmall.cart.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 RemoteCallConfig {
  7.     @Bean
  8.     public RestTemplate restTemplate() {
  9.         return new RestTemplate();
  10.     }
  11. }
复制代码
远程调用

接下来,我们修改cart-service中的com.hmall.cart.service.impl.CartServiceImpl的handleCartItems方法,发送http哀求到item-service:

可以看到,使用RestTemplate发送http哀求与前端ajax发送哀求非常相似,都包含四部分信息:


  • ① 哀求方式
  • ② 哀求路径
  • ③ 哀求参数
  • ④ 返回值类型
handleCartItems方法的完整代码如下:
  1. private void handleCartItems(List<CartVO> vos) {
  2.     // TODO 1.获取商品id
  3.     Set<Long> itemIds = vos.stream().map(CartVO::getItemId).collect(Collectors.toSet());
  4.     // 2.查询商品
  5.     // List<ItemDTO> items = itemService.queryItemByIds(itemIds);
  6.     // 2.1.利用RestTemplate发起http请求,得到http的响应
  7.     ResponseEntity<List<ItemDTO>> response = restTemplate.exchange(
  8.             "http://localhost:8081/items?ids={ids}",
  9.             HttpMethod.GET,
  10.             null,
  11.             new ParameterizedTypeReference<List<ItemDTO>>() {
  12.             },
  13.             Map.of("ids", CollUtil.join(itemIds, ","))
  14.     );
  15.     // 2.2.解析响应,检查HTTP响应的状态码是否在200-299范围内,表示请求成功。
  16.     if(!response.getStatusCode().is2xxSuccessful()){
  17.         // 查询失败,直接结束
  18.         return;
  19.     }
  20.     // 获取响应体中的商品信息
  21.     List<ItemDTO> items = response.getBody();
  22.     if (CollUtils.isEmpty(items)) {
  23.         return;
  24.     }
  25.     // 3.转为 id 到 item的map
  26.     Map<Long, ItemDTO> itemMap = items.stream().collect(Collectors.toMap(ItemDTO::getId, Function.identity()));
  27.     // 4.写入vo
  28.     for (CartVO v : vos) {
  29.         ItemDTO item = itemMap.get(v.getItemId());
  30.         if (item == null) {
  31.             continue;
  32.         }
  33.         v.setNewPrice(item.getPrice());
  34.         v.setStatus(item.getStatus());
  35.         v.setStock(item.getStock());
  36.     }
  37. }
复制代码
好了,如今重启cart-service,再次测试查询我的购物车列表接口:

可以发现,全部商品干系数据都已经查询到了。
在这个过程中,item-service提供了查询接口,cart-service使用Http哀求调用该接口。因此item-service可以称为服务的提供者,而cart-service则称为服务的斲丧者或服务调用者。
总结

什么时间需要拆分微服务?


  • 如果是创业型公司,最好先用单体架构快速迭代开辟,验证市场运作模子,快速试错。当业务跑通以后,随着业务规模扩大、人员规模增加,再考虑拆分微服务。
  • 如果是大型企业,有充足的资源,可以在项目开始之初就搭建微服务架构。
如何拆分?


  • 首先要做到高内聚、低耦合
  • 从拆分方式来说,有横向拆分和纵向拆分两种。纵向就是按照业务功能模块,横向则是拆分通用性业务,提高复用性
服务拆分之后,不可避免的会出现跨微服务的业务,此时微服务之间就需要举行远程调用。微服务之间的远程调用被称为RPC,即远程过程调用。RPC的实现方式有许多,比如:


  • 基于Http协议
  • 基于Dubbo协议
我们讲堂中使用的是Http方式,这种方式不关心服务提供者的具体技术实现,只要对外暴露Http接口即可,更符合微服务的需要。
Java发送http哀求可以使用Spring提供的RestTemplate,使用的基本步骤如下:


  • 注册RestTemplate到Spring容器
  • 调用RestTemplate的API发送哀求,常见方法有:

    • getForObject:发送Get哀求并返回指定类型对象
    • PostForObject:发送Post哀求并返回指定类型对象
    • put:发送PUT哀求
    • delete:发送Delete哀求
    • exchange:发送任意类型哀求,返回ResponseEntity


免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?立即注册

x
回复

使用道具 举报

0 个回复

倒序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

张裕

金牌会员
这个人很懒什么都没写!

标签云

快速回复 返回顶部 返回列表