谷粒商城分布式基础&高级篇总结

谷粒商城分布式基础&高级篇总结

前言

本文主要是对谷粒商城分布式基础和高级篇的技术栈总结。

1 分布式基础篇总结

1671954399556


  1. 分布式基础概念

微服务

体现独立自治:为每一个不同功能的项目都创建了属于它自己的服务;在实际开发中,这些不同功能的项目可以交给不同的开发人员进行并行开发,继而可以提高项目的开发速度。
有了微服务之后,有牵扯到了其他问题。

注册中心

如果我们的微服务之间不进行互相调用,彼此之间都没有关系,那也就不需要这个注册中心,可是在商品系统中,我们可能会调用会员服务,包括可能会调用库存的一些服务,那这时就需要一个注册中心,来实时感知每一个微服务都在哪个位置;这样服务之间想要互相调用,只需要从注册中心获取到地址列表,从而给相应的微服务发送请求就行了。

配置中心

无论是单体应用还是微服务,我们都需要一个配置中心,【好处就是服务上线以后,我们不需要修改本地源代码的配置,继而打包之后再发布,而是在线上通过一个可视化的界面修改配置,修改完之后,服务直接使用的最新更新了的配置。

远程调用

在开发期间,我们发现了很多远程调用的场景,我们使用feign来做远程调用。在springcloud中,就是使用feign给对方服务发送一个请求而已:【以前通过Ajax发送请求;通过超链接点击跳转到另外一个地址,其实这些都是发送请求来要数据】;而在微服务开发期间,我们想要别的服务的数据,我们就可以使用feign来发请求。

使用feign流程:导入依赖、开启远程调用功能@EnableFeginClients、编写一个接口、指定我们要调用那个服务的那个请求就行了【因为远程传输期间都是JSON数据互相发送,所以接口中方法的类型只需要写能兼容JSON数据的类型就行】
前提是这些微服务都需要注册到注册中心:通过配置注册中心的地址、以及我们当前应用的名字即可:因为我们的注册中心、配置中心都是使用的nacos,所以简化了很多的配置,此外最重要的还有需要开启服务注册发现注解:@EnableDiscoveryClient。】这是feign想要使用的前提,微服务都得注册进来。

网关

我们所有的请求都是发送给网关,由网关代理给其他服务。对网关就需要做一些统一的功能设置:比如我们在前期开发过程中,做了一个全局配置【统一的跨域配置GulimallCorsConfiguration】:前端请求给后台发送Ajax请求的时候,由于我们的地址不一样,所以会引起跨域,所以我们在网关处统一配置跨域,这样我们就不需要在微服务的各个位置来编写跨域配置了。

  1. 基础开发

SpringBoot2.0

在基础开发中,我们使用springboot2.0,在开发中我们可能没有觉得springboot2.0有什么重大的变化,因为现在我们还没有用到,除了发现在springboot2.0中有些配置有可能有些变化,【比如在配置文件中,在配置一些相关配置的时候,可能和之前有些变化,继而我们可能没有感觉springboot2.0有什么变化】
实际上,springboot2.0基于spring5做了一个最大的变化,引入了Reactor:反应式编程,它给我们带来了web开发中的webFlux【可以很容易的创建出一个高性能、高并发的web应用】,现在我们还没用到,在高级部分我们就会转入webFlux的开发模式。现在只在网关配置跨域的使用用到了webflux,那就是跨域的这个filter网关,【CorsWebFilter】,它是属于webflux编程模式的,但是在这里使用我们也没有感觉出webflux编程有什么不同;但是在这个filter的源码中,可以看到它返回的是Mono

SpringCloud

对于springcloud我们这里也只是简单的使用了几个注解:比如开启服务注册发现功能@EnableDiscoveryClient ,开启feign的远程调用@EnableFeignClients 、顶多编写了feign的几个远程接口。

mybatisplus
我们也只是用mybatis最原生的方式,配置了简单的包扫描@MapperScan、此外mapper接口都是自动生成的。【一些自定义的方法我们只需要在mybatis的配置文件中配置即可】。

vue组件化开发
在基础开发部分,大量时间使用了vue组件化开发,比如商品系统的分类维护、品牌管理、包括平台属性的一些 功能:如属性分组等,让大家对整个vue的开发有了入门、相当于带大家做了一个基本的全栈开发。

阿里云对象存储

体会了一下调用第三方服务应该怎么做。后面如果我们想要使用第三方服务,可能都需要申请开通,调用权限、调用人家SDK提供的各种功能。【就是对照人家的SDK开发文档进行开发即可】

  1. 环境

使用Linux + Docker的方式:使用docker部署MySQL、Redis【暂时没有用到】
在创建虚拟机的时候使用了一个非常快的方式:vagrant【只需要一个命令:vagrant init centos7】即可创建一个虚拟机;启动虚拟机【vagrant up】;想要连接虚拟机,只需要在Windows的cmd命令行窗口【vagrant ssh】;
其中最为方便的就是人人开源为我们提供的逆向工程【它的逆向工程其实就是通过MyBatisPlus的逆向工程的一个改版】,但是它为我们更多的编写了controller接口、以及vue的基础组件。有了它,基本的crud,从controller层、到service层、到dao层、乃至vue页面层我们都不需要开发,所以大家要小心了,基础的crud程序员以后可能再也不需要了。我们要做的是高级部分的高并发功能。

  1. 开发规范

开发规范:
在开发期间,我们每一个数据的提交新增,都需要用到数据校验,比如Jsr303 ;虽然我们只用了一个场景,其他嫌太麻烦都没写,但是直接往上套就没问题。
包括全局异常处理、全局统一返回【系统中的R对象】、网关做了全局的跨域处理。以及将我们系统中的一些状态通过枚举来写好,将业务状态码都抽取成了相应的枚举状态;包括在开发期间,vo、to以及其他各种o的划分【分包与处理】:以后我们开发期间,凡是收页面的请求,我们都应该来开发相应的vo。包括使用mybatisplus提供的逻辑删除功能,以及lombok为我们提供的两个简单注解:一个@Data【为我们生成getter、setter方法】,一个是 @Slf4j【为我们提供了简单的日志对象】,我们只需要使用log.error/log.info即可使用。


2 高级篇总结

基础篇注重的是 基本的增删改查的能力;搭建的是一个前后分离的环境来做整个后台管理系统的增删改查,其中穿插了一些技术:数据校验、对象存储、一些VO数据的封装等等...

高级篇:实现的是整个商城功能,而不是后台管理系统;其中业务有购物车、结账、详情、检索;将所有的功能都抽取成了一个微服务:也就是说在整个商城系统中,将每个业务拆分成了微服务,由许多微服务共同组合成商城;其中用到的技术和难点:分布式开发期间:核心掌握 SpringCloud组件:包括SpringCloud Alibaba、SpringCloud。

在Springcloud组件中使用的最为频繁的就是 feign远程调用:开启接口,声明feign客户端&&复制目标方法及其签名。

  • 课件详解

    ![1671895330266](F:\JAVA listen\尚硅谷Java学科全套教程(总207.77GB)\谷粒商城\课件(最新)\自己\谷粒商城之高级篇(2)\1671895330266.png)

01 响应式编程

只在最后网关SentinelGatewayConfig类中有使用:网关的sentinel的容错回调中使用了一下;

![1671895266958](F:\JAVA listen\尚硅谷Java学科全套教程(总207.77GB)\谷粒商城\课件(最新)\自己\谷粒商城之高级篇(2)\1671895266958.png)

02 接口幂等性

分布式开发中最需要关注的就是接口幂等性问题:无论服务怎么调用,都希望接口是幂等的;这样我们无论调用多少次,结果都是一样的;这个是我们分布式系统里面业务功能接口的保证可以通过加锁、数据库的乐观锁、悲观锁字段、加事务等都可以实现接口幂等性;我们在系统中使用最多的就是令牌机制【类似于验证码】:我给你发一个令牌,你下一次给我带上,用过一次就销毁。

03 事务

在分布式系统中,除了每个系统要解决自己的事务问题之外,分布式系统中调用还要有分布式事务的存在;在这个过程中,还顺带讲解了 springcloud alibaba 中的 seata组件:它可以解决分布式事务,但是它最好用的地方在于 后台管理系统中使用:比如我们在后台管理系统中添加 一个商品,保存的时候需要将 它的优惠信息、库存等在系统中保存,要成功都成功,要失败都失败,【对于并发性能要求不高的,例如后台管理系统,我们可以使用seata 来做分布式事务】;
但是对于高并发系统,我们使用的是最终一致性,所以分布式事务的最终解决方案就是 RabbitMQ:首先发送一个消息:【可靠消息的保证:发送端和消费端的两端确认机制】,将消息发送出去后,最终的系统只需要监听这些消息,完了之后根据消息改变我们系统的数据;【不能达到强一致,达到的是柔性事务的最终一致:最终看到的数据是我们想要的就行。】

04 性能与压力测试

对于高并发系统,使用Jmeter进行压力测试;在压测期间,还可以监控JVM的性能,【使用jvisualvm控制台,或者Jconsole】。

05 缓存和分布式锁
特别在分布式系统中,我们要保证接口的吞吐量,性能的提升【缓存是必须的】,所以在高并发系统中,缓存是一个很重要的提升性能的手段,但是使用缓存期间,也有一些问题:缓存的击穿【缓存的单点问题】,缓存雪崩【缓存的大面积问题】,缓存穿透【一直从数据库查询null值】:解决方案:使用分布式

当大并发量全部要来查询数据库,大家先来查询缓存,使用分布式锁来锁住,只有一个请求进来,查询不到缓存再来查询数据库,最终的效果就是只会给数据库放一条查询,而其他人得到锁之后,下次只会来查询缓存,也就不用再去查询数据库了,【分布式锁除了缓存问题使用外,接口幂等性的保证也可以使用】
比如现在有 几个并发的请求,都来调用了一个接口,这个接口只能调用一次,这一次数据是怎么样,那就是怎么样:我们释放锁之后,如果再来调用,就判断一下数据的状态,如果已经被处理了,那就不再处理。
定时任务中也有使用:比如定时启动了3台机器,这3台机器定时同时启动了一个任务,比如都来上架我们的秒杀商品,只要我们使用了分布式锁,我们上架过了,就不需要再次 上架;分布式锁可以来锁住我们所有的机器,让
一个机器来执行这个事情即可。

06 ElasticSearch

我们商品的检索肯定不能放在mysql中进行,通过写一些SQL执行,这样性能会大幅度下降;我们的检索功能:无论是分类检索,还是按照名字检索;所有的检索数据都保存在ElasticSearch中【一个非常专业的检索引擎】,注意它的安装使用都需要非常清楚。

07 异步和线程池

在高并发系统中,异步是非常必须的:我们复习了以前那种简单的 new Thread start这种简单的异步,如果在高并发系统中,我们每一个请求进来都new Thread start,这样资源将很快耗尽;所以为了控制住系统的资源,我们使用线程池:以后所有的异步任务都要提交给线程池;这样的话,线程池中就有一个最大量,比如核心是20个线程,最大是200个,排队可以排500个、800个、乃至更多,我们通过线程池控制住资源,峰值也就是200个正在运行的,以及800个正在排队的,再也不可能占用更多的资源。这样有了资源的统一管控后,我们就不用害怕因为系统的某些资源耗尽导致系统崩溃
我们使用异步,将任务提交给线程池,但是可能异步任务中有顺序,我们可以使用异步编排CompetableFuture

08 单点登录和社交登录

我们着重演示了使用微博进行社交登录,此外还演示了单点登录,比如我们在不同域名下如何登录。但是我们后来是相同域名的,所以我们暂时可以不用使用单点登录。
但是在相同域名下,登录一处,在处处都可以实现单点登录效果,这时我们就整合了 spring session:一处登录,处处可用。【将所有微服务的session进行了同步,只要登录成功之后,去任何微服务,都可以获取到session】---> 使用spring session 解决了分布式系统session不一致问题。

09 商城业务

特别是购物车、订单、秒杀等,所有的业务都做了实现,但是一些细节需要我们自己填充。核心业务:商品上架【后台管理系统】、商品检索、商品详情、购物车、订单、秒杀。其中在商品详情里面,使用最多的是缓存技术:除了整合redis,还整合了springCache

整合springCache来方便的使用缓存,以后我们的全系统都应该使用这种方式来进行缓存。
缓存出现的不一致问题该如何解决缓存的清空以及缓存的更新,使用spring cache 都可以很方便的解决这些问题。

10 RabbitMQ

做分布式事务,来实现最终一致性的时候,rabbitmq是一个非常必要的工具。
我们只需要给A服务给B服务发送一个消息,A服务不用关心怎么做。【在订单系统中,我们使用rabbitmq来完成分布式事务】;【在秒杀系统中,加入rabbitmq队列来进行削峰处理:将所有的流量排队放到队列中,由后台慢慢的进行消费】。此外,通过rabbitmq,A服务和B服务不用关心接口调用了,A都不需要调用B,相当于我们将应用之间进行了解耦。
在订单的关单中关于rabbitmq的使用:我们通过死信队列,保证关单的数据都能被关掉。

11 支付

我们整合了支付宝的沙箱来进行支付。

12 定时任务与分布式调度

我们秒杀系统的所有上架,都需要定时任务来做

13 ShardingSphere

对于 13 分库分表 ShardingSphere在高可用做mysql集群时在使用。

14 SpringCloud组件

SpringCloud中的组件都需要进行掌握:除了nacos作为注册中心我们在一直使用外,配置中心只进行了演示;以后我们需要将所有微服务的配置都放给配置中心,【通过nacos的控制台也可以修改配置,修改完了之后微服务可以在不下线的情况下,应用到最新的配置】
最后,我们为了保护整个系统,引入了 sentinel【作为一个流量哨兵】:从流量入口开始,保护我们任何想要保护的资源;当然,我们最好结合 Sleuth + Zipkin进行服务链路追踪,这样可以看到我们整个系统的运行状况,链路追踪。可以找出一些异常问题,通过sentinel流量哨兵进行服务的降级,保护整个系统。

技术中最重要的是使用缓存来加快速度。异步也是来加快速度和控制资源。此外通过消息队列,无论是流量削峰,还是分布式做最终一致也好,我们都可以通过消息队列来减轻某一个服务的压力【所有消息存入队列中,闲的时候再来慢慢消费】。

![1671896555408](F:\JAVA listen\尚硅谷Java学科全套教程(总207.77GB)\谷粒商城\课件(最新)\自己\谷粒商城之高级篇(2)\1671896555408.png)

在高级篇构建一个高并发系统,除了引入springcloud组件或者是SpringCloud Alibaba来作为周边设施外,高并发有三宝:
缓存、异步、队排好

缓存就是使用redis作为缓存,保证它的缓存一致性,保证缓存的击穿、穿透等这些问题不会出现;

异步结合线程池,不能只是new Tread start;结合线程池以及异步编排来作为整个异步功能。

队排好:使用rabbitmq将所有的高峰流量,或者要做分布式事务的这些消息,全都放进rabbitmq消息队列中,让他们排好队,后台服务一个个处理。

加上这3个手段,构建高并发系统并不难。

posted @ 2022-12-26 21:37  wylja  阅读(678)  评论(0编辑  收藏  举报