老板要搞微服务,只能硬着头皮上了...

百家 作者:51CTO技术栈 2020-06-19 19:08:27

微服务越来越火。很多互联网公司,甚至一些传统行业的系统都采用了微服务架构。


图片来自 Pexels


体会到微服务带来好处的同时,很多公司也明显感受到微服务化带来的一系列让人头疼的问题。


本文是笔者对自己多年微服务化经历的总结。如果你正准备做微服务转型,或者在微服务化过程中遇到了困难。此文很可能会帮到你!


正文开始前,为了让各位读友更好的理解本文内容,先花两分钟了解一下微服务的优缺点。


聊起微服务,很多朋友都了解微服务带来的好处,罗列几点:

  • 模块化,降低耦合。将单体应用按业务模块拆分成多个服务,如果某个功能需要改动,大多数情况,我们只需要弄清楚并改动对应的服务即可。

    只改动一小部分就能满足要求,降低了其他业务模块受影响的可能性。从而降低了业务模块间的耦合性。

  • 屏蔽与自身业务无关技术细节。例如,很多业务需要查询用户信息,在单体应用的情况下,所有业务场景都通过 DAO 去查询用户信息,随着业务发展,并发量增加,用户信息需要加缓存。

    这样所有业务场景都需要关注缓存,微服务化之后,缓存由各自服务维护,其他服务调用相关服务即可,不需要关注类似的缓存问题。

  • 数据隔离,避免不同业务模块间的数据耦合。不同的服务对应不同数据库表,服务之间通过服务调用的方式来获取数据。

  • 业务边界清晰,代码边界清晰。单体架构中不同的业务,代码耦合严重,随着业务量增长,业务复杂后,一个小功能点的修改就可能影响到其他业务点,开发质量不可控,测试需要回归,成本持续提高。

  • 显著减少代码冲突。在单体应用中,很多人在同一个工程上开发,会有大量的代码冲突问题。微服务化后,按业务模块拆分成多个服务,每个服务由专人负责,有效减少代码冲突问题。

  • 可复用,显著减少代码拷贝现象。


微服务确实带来不少好处,那么微服务有没有什么问题呢?答案是肯定的!

例如:
  • 微服务系统稳定性问题。微服务化后服务数量大幅增加,一个服务故障就可能引发大面积系统故障。比如服务雪崩,连锁故障。当一个服务故障后,依赖他的服务受到牵连也发生故障。

  • 服务调用关系错综复杂,链路过长,问题难定位。微服务化后,服务数量剧增,大量的服务管理起来会变的更加复杂。由于调用链路变长,定位问题也会更加困难。

  • 数据一致性问题。微服务化后单体系统被拆分成多个服务,各服务访问自己的数据库。而我们的一次请求操作很可能要跨多个服务,同时要操作多个数据库的数据,我们发现以前用的数据库事务不好用了。跨服务的数据一致性和数据完整性问题也就随之而来了。

  • 微服务化过程中,用户无感知数据库拆分、数据迁移的挑战。


如何保障微服务系统稳定性?


互联网系统为大量的 C 端用户提供服务,如果隔三差五的出问题宕机,会严重影响用户体验,甚至导致用户流失。所以稳定性对互联网系统非常重要!


接下来笔者根据自己的实际经验来聊聊基于微服务的互联网系统的稳定性。

①雪崩效应产生原因,如何避免?


微服务化后,服务变多,调用链路变长,如果一个调用链上某个服务节点出问题,很可能引发整个调用链路崩溃,也就是所谓的雪崩效应。

举个例子,详细理解一下雪崩。如上图,现在有 A,B,C 三个服务,A 调 B,B 调 C。

假如 C 发生故障,B 方法 1 调用 C 方法 1 的请求不能及时返回,B 的线程会发生阻塞等待。


B 会在一定时间后因为线程阻塞耗尽线程池所有线程,这时 B 就会无法响应 A 的请求。


A 调用 B 的请求不能及时返回,A 的线程池线程资源也会逐渐被耗尽,最终 A 也无法对外提供服务。这样就引发了连锁故障,发生了雪崩。

纵向:C 故障引发 B 故障,B 故障引发 A 故障,最终发生连锁故障。横向:方法 1 出问题,导致线程阻塞,进而线程池线程资源耗尽,最终服务内所有方法都无法访问,这就是“线程池污染”。


为了避免雪崩效应,我们可以从两个方面考虑:

在服务间加熔断:解决服务间纵向连锁故障问题。比如在 A 服务加熔断,当 B 故障时,开启熔断,A 调用 B 的请求不再发送到 B,直接快速返回。这样就避免了线程等待的问题。


当然快速返回什么,Fallback 方案是什么,也需要根据具体场景,比如返回默认值或者调用其他备用服务接口。


如果你的场景适合异步通信,可以采用消息队列,这样也可以有效避免同步调用的线程等待问题。

服务内(JVM 内)线程隔离:解决横向线程池污染的问题。为了避免因为一个方法出问题导致线程等待最终引发线程资源耗尽的问题,我们可以对 Tomcat,Dubbo 等的线程池分成多个小线程组,每个线程组服务于不同的类或方法。


一个方法出问题,只影响自己不影响其他方法和类。常用开源熔断隔离组件:Hystrix,Resilience4j。


②如何应对突发流量对服务的巨大压力?

促销活动或秒杀时,访问量往往会猛增数倍。技术团队在活动开始前一般都会根据预估访问量适当增加节点,但是假如流量预估少了(实际访问量远大于预估的访问量),系统就可能会被压垮。

所以我们可以在网关层(Zuul,Gateway,Nginx 等)做限流,如果访问量超出系统承载能力,就按照一定策略抛弃超出阈值的访问请求(也要注意用户体验,可以给用户返回一个友好的页面提示)。


可以从全局,IP,userID 等多维度做限流。限流的两个主要目的:
  • 应对突发流量,避免系统被压垮(全局限流和 IP 限流)

  • 防刷,防止机器人脚本等频繁调用服务(userID 限流和 IP 限流)


③数据冗余

在核心链路上,服务可以冗余它依赖的服务的数据,依赖的服务故障时,服务尽量做到自保。


比如订单服务依赖库存服务。我们可以在订单服务冗余库存数据(注意控制合理的安全库存,防超卖)。


下单减库存时,如果库存服务挂了,我们可以直接从订单服务取库存。可以结合熔断一起使用,作为熔断的 Fallback(后备)方案。

④服务降级


可能很多人都听过服务降级,但是又不知道降级是怎么回事。实际上,上面说的熔断,限流,数据冗余,都属于服务降级的范畴。


还有手动降级的例子,比如大促期间我们会关掉第三方物流接口,页面上也关掉物流查询功能,避免拖垮自己的服务。


这种降级的例子很多。不管什么降级方式,目的都是让系统可用性更高,容错能力更强,更稳定。关于服务降级详见本文后面的内容。

⑤缓存要注意什么?

主要有如下三点:


缓存穿透:对于数据库中根本不存在的值,请求缓存时要在缓存记录一个空值,避免每次请求都打到数据库


缓存雪崩:在某一时间缓存数据集中失效,导致大量请求穿透到数据库,将数据库压垮。


可以在初始化数据时,差异化各个 key 的缓存失效时间,失效时间=一个较大的固定值+较小的随机值。


缓存热点。有些热点数据访问量会特别大,单个缓存节点(例如 Redis)无法支撑这么大的访问量。


如果是读请求访问量大,可以考虑读写分离,一主多从的方案,用从节点分摊读流量;如果是写请求访问量大,可以采用集群分片方案,用分片分摊写流量。


以秒杀扣减库存为例,假如秒杀库存是 100,可以分成 5 片,每片存 20 个库存。


⑥关于隔离的考虑


需要考虑如下几点:


部署隔离:我们经常会遇到秒杀业务和日常业务依赖同一个服务,以及 C 端服务和内部运营系统依赖同一个服务的情况,比如说都依赖订单服务。


而秒杀系统的瞬间访问量很高,可能会对服务带来巨大的压力,甚至压垮服务。内部运营系统也经常有批量数据导出的操作,同样会给服务带来一定的压力。


这些都是不稳定因素。所以我们可以将这些共同依赖的服务分组部署,不同的分组服务于不同的业务,避免相互干扰。


数据隔离:极端情况下还需要缓存隔离,数据库隔离。以秒杀为例,库存和订单的缓存(Redis)和数据库需要单独部署!


数据隔离后,秒杀订单和日常订单不在相同的数据库,之后的订单查询怎么展示?可以采用相应的数据同步策略。


比如,在创建秒杀订单后发消息到消息队列,日常订单服务收到消息后将订单写入日常订单库。注意,要考虑数据的一致性,可以使用事务型消息。


业务隔离:还是以秒杀为例。从业务上把秒杀和日常的售卖区分开来,把秒杀做为营销活动,要参与秒杀的商品需要提前报名参加活动,这样我们就能提前知道哪些商家哪些商品要参与秒杀。


可以根据提报的商品提前生成商品详情静态页面并上传到 CDN 预热,提报的商品库存也需要提前预热,可以将商品库存在活动开始前预热到 Redis,避免秒杀开始后大量访问穿透到数据库。

    

⑦CI 测试&性能测试


CI 测试,持续集成测试,在我们每次提交代码到发布分支前自动构建项目并执行所有测试用例,如果有测试用例执行失败,拒绝将代码合并到发布分支,本次集成失败。CI 测试可以保证上线质量,适用于用例不会经常变化的稳定业务。


性能测试,为了保证上线性能,所有用户侧功能需要进行性能测试。上线前要保证性能测试通过。而且要定期做全链路压测,有性能问题可以及时发现。

⑧监控


我们需要一套完善的监控系统,系统出问题时能够快速告警,最好是系统出问题前能提前预警。


包括系统监控(CPU,内存,网络 IO,带宽等监控),数据库监控(QPS,TPS,慢查询,大结果集等监控),缓存中间件监控(如 Redis),JVM 监控(堆内存,GC,线程等监控),全链路监控(pinpoint,skywaking,cat等),各种接口监控(QPS,TPS 等)

⑨CDN


可以充分利用 CDN。除了提高用户访问速度之外,页面静态化之后存放到 CDN,用 CDN 扛流量,可以大幅减少系统(源站)的访问压力。同时也减少了网站带宽压力。对系统稳定性非常有好处。


⑩避免单点问题


除了服务要多点部署外,网关,数据库,缓存也要避免单点问题,至少要有一个 Backup,而且要可以自动发现上线节点和自动摘除下线和故障节点。

⑪网络带宽

避免带宽成为瓶颈,促销和秒杀开始前提前申请带宽。不光要考虑外网带宽,还要考虑内网带宽,有些旧服务器网口是千兆网口,访问量高时很可能会打满。


此外,一套完善的灰度发布系统,可以让上线更加平滑,避免上线大面积故障。DevOps 工具,CI,CD 对系统稳定性也有很大意义。


关于服务降级


提起服务降级,估计很多人都听说过,但是又因为亲身经历不多,所以可能不是很理解。下面结合具体实例从多方面详细阐述服务降级。


互联网分布式系统中,经常会有一些异常状况导致服务器压力剧增,比如促销活动时访问量会暴增,为了保证系统核心功能的稳定性和可用性,我们需要一些应对策略。


这些应对策略也就是所谓的服务降级。下面根据笔者的实际经历,跟大家聊聊服务降级那些事儿。希望对大家有所启发!

①关闭次要功能


在服务压力过大时,关闭非核心功能,避免核心功能被拖垮。


例如,电商平台基本都支持物流查询功能,而物流查询往往要依赖第三方物流公司的系统接口。


物流公司的系统性能往往不会太好。所以我们经常会在双 11 这种大型促销活动期间把物流接口屏蔽掉,在页面上也关掉物流查询功能。这样就避免了我们自己的服务被拖垮,也保证了重要功能的正常运行。

②降低一致性之读降级


对于读一致性要求不高的场景。在服务和数据库压力过大时,可以不读数据库,降级为只读缓存数据。以这种方式来减小数据库压力,提高服务的吞吐量。


例如,我们会把商品评论评价信息缓存在 Redis 中。在服务和数据库压力过大时,只读缓存中的评论评价数据,不在缓存中的数据不展示给用户。


当然评论评价这种不是很重要的数据可以考虑用 NOSQL 数据库存储,不过我们曾经确实用 MySQL 数据库存储过评论评价数据。

③降低一致性之写入降级

在服务压力过大时,可以将同步调用改为异步消息队列方式,来减小服务压力并提高吞吐量。


既然把同步改成了异步也就意味着降低了数据一致性,保证数据最终一致即可。


例如,秒杀场景瞬间生成订单量很高。我们可以采取异步批量写数据库的方式,来减少数据库访问频次,进而降低数据库的写入压力。


详细步骤:后端服务接到下单请求,直接放进消息队列,消费端服务取出订单消息后,先将订单信息写入 Redis,每隔 100ms 或者积攒 100 条订单,批量写入数据库一次。


前端页面下单后定时向后端拉取订单信息,获取到订单信息后跳转到支付页面。用这种异步批量写入数据库的方式大幅减少了数据库写入频次,从而明显降低了订单数据库写入压力。


不过,因为订单是异步写入数据库的,就会存在数据库订单和相应库存数据暂时不一致的情况,以及用户下单后不能及时查到订单的情况。


因为是降级方案,可以适当降低用户体验,所以我们保证数据最终一致即可。流程如下图:

④屏蔽写入


很多高并发场景下,查询请求都会走缓存,这时数据库的压力主要是写入压力。所以对于某些不重要的服务,在服务和数据库压力过大时,可以关闭写入功能,只保留查询功能。这样可以明显减小数据库压力。


例如,商品的评论评价功能。为了减小压力,大促前可以关闭评论评价功能,关闭写接口,用户只能查看评论评价。而大部分查询请求走查询缓存,从而大幅减小数据库和服务的访问压力。

⑤数据冗余


服务调用者可以冗余它所依赖服务的数据。当依赖的服务故障时,服务调用者可以直接使用冗余数据。


例如,我之前在某家自营电商公司。当时的商品服务依赖于价格服务,获取商品信息时,商品服务要调用价格服务获取商品价格。


因为是自营电商,商品和 SKU 数量都不太多,一两万的样子。所以我们在商品服务冗余了价格数据。当价格服务故障后,商品服务还可以从自己冗余的数据中取到价格。


当然这样做价格有可能不是最新的,但毕竟这是降级方案,牺牲一些数据准确性,换来系统的可用性还是很有意义的!


注:由于一个商品会有多个价格,比如普通价,会员价,促销直降价,促销满减价,所以我们把价格做成了单独的服务。


数据冗余可以结合熔断一起使用,实现自动降级。下面的熔断部分会详细说明。


⑥熔断和 Fallback


熔断是一种自动降级手段。当服务不可用时,用来避免连锁故障,雪崩效应。发生在服务调用的时候,在调用方做熔断处理。


熔断的意义在于,调用方快速失败(Fail Fast),避免请求大量阻塞。并且保护被调用方。


详细解释一下,假设 A 服务调用 B 服务,B 发生故障后,A 开启熔断:
  • 对于调用方 A:请求在 A 直接快速返回,快速失败,不再发送到 B。 避免因为 B 故障,导致 A 的请求线程持续等待,进而导致线程池线程和 CPU 资源耗尽,最终导致 A 无法响应,甚至整条调用链故障。

  • 对于被调用方 B:熔断后,请求被 A 拦截,不再发送到 B,B 压力得到缓解,避免了仍旧存活的 B 被压垮,B 得到了保护。


还是以电商的商品和价格服务为例。获取商品信息时,商品服务要调用价格服务获取商品价格。为了提高系统稳定性,我们要求各个服务要尽量自保。


所以我们在商品服务加了熔断,当价格服务故障时,商品服务请求能够快速失败返回,保证商品服务不被拖垮,进而避免连锁故障。


看到这,可能有读者会问,快速失败后价格怎么返回呢?因为是自营电商,商品和 SKU 数量都不太多,一两万的样子。所以我们做了数据冗余,在商品服务冗余了价格数据。

这样我们在熔断后获取价格的 Fallback 方案就变成了从商品服务冗余的数据去取价格。


下图为商品服务熔断关闭和开启的对比图:

开源熔断组件:Hystrix,Resilience4j 等。

⑦限流


说起服务降级,就不可避免的要聊到限流。我们先考虑一个场景,例如电商平台要搞促销活动,我们按照预估的峰值访问量,准备了 30 台机器。


但是活动开始后,实际参加的人数比预估的人数翻了 5 倍,这就远远超出了我们的服务处理能力,给后端服务、缓存、数据库等带来巨大的压力。


随着访问请求的不断涌入,最终很可能造成平台系统崩溃。对于这种突发流量,我们可以通过限流来保护后端服务。


因为促销活动流量来自于用户,用户的请求会先经过网关层再到后端服务,所以网关层是最合适的限流位置,如下图:

另外,考虑到用户体验问题,我们还需要相应的限流页面。当某些用户的请求被限流拦截后,把限流页面返回给用户。页面如下图:

另外一个场景,假如有一个核心服务,有几十个服务都调用他。如果其中一个服务调用者出了 Bug,频繁调用这个核心服务,可能给这个核心服务造成非常大的压力,甚至导致这个核心服务无法响应。


同时也会影响到调用他的几十个服务。所以每个服务也要根据自己的处理能力对调用者进行限制。


对于服务层的限流,我们一般可以利用 Spring AOP,以拦截器的方式做限流处理。这种做法虽然可以解决问题,但是问题也比较多。

比如一个服务中有 100 个接口需要限流,我们就要写 100 个拦截器。而且限流阈值经常需要调整,又涉及到动态修改的问题。


为了应对这些问题,很多公司会有专门的限流平台,新增限流接口和阈值变动可以直接在限流平台上配置。


关于限流,还有很多细节需要考虑,比如限流算法、毛刺现象等。篇幅原因,这些问题就不在本文讨论了。


开源网关组件:Nginx,Zuul,Gateway,阿里 Sentinel 等。

⑧服务降级总结和思考


上面我们结合具体案例解释了多种降级方式。实际上,关于服务降级的方式和策略,并没有什么定式,也没有标准可言。


上面的降级方式也没有涵盖所有的情况。不同公司不同平台的做法也不完全一样。


不过,所有的降级方案都要以满足业务需求为前提,都是为了提高系统的可用性,保证核心功能正常运行。

⑨降级分类


一般我们可以把服务降级分为手动和自动两类。手动降级应用较多,可以通过开关的方式开启或关闭降级。


自动降级,比如熔断和限流等属于自动降级的范畴。大多手动降级也可以做成自动的方式,可以根据各种系统指标设定合理阈值,在相应指标达到阈值上限自动开启降级。


在很多场景下,由于业务过于复杂,需要参考的指标太多,自动降级实现起来难度会比较大,而且也很容易出错。


所以在考虑做自动降级之前一定要充分做好评估,相应的自动降级方案也要考虑周全。

⑩大规模分布式系统如何降级?


在大规模分布式系统中,经常会有成百上千的服务。在大促前往往会根据业务的重要程度和业务间的关系批量降级。


这就需要技术和产品提前对业务和系统进行梳理,根据梳理结果确定哪些服务可以降级,哪些服务不可以降级,降级策略是什么,降级顺序怎么样。


大型互联网公司基本都会有自己的降级平台,大部分降级都在平台上操作,比如手动降级开关,批量降级顺序管理,熔断阈值动态设置,限流阈值动态设置等等。


本节的主要目的是通过具体实例,让大家了解服务降级,并提供一些降级的思路。具体的降级方式和方案还是要取决于实际的业务场景和系统状况。


微服务架构下数据一致性问题


服务化后单体系统被拆分成多个服务,各服务访问自己的数据库。而我们的一次请求操作很可能要跨多个服务,同时要操作多个数据库的数据,我们发现以前用的数据库事务不好用了。那么基于微服务的架构如何保证数据一致性呢?


好,咱们这次就盘一盘分布式事务,最终一致,补偿机制,事务型消息!


提起这些,大家可能会想到两阶段提交,XA,TCC,Saga,还有最近阿里开源的 Seata(Fescar),这些概念网上一大堆文章,不过都太泛泛,不接地气,让人看了云里雾里。

我们以 TCC 分布式事务和 RocketMQ 事务型消息为例,做详细分享!这个弄明白了,也就清楚分布式事务,最终一致,补偿机制这些概念啦!


①TCC 分布式事务


TCC(Try-Confirm-Cancel)是分布式事务的一种模式,可以保证不同服务的数据最终一致。


目前有不少 TCC 开源框架,比如 Hmily,ByteTCC,TCC-Transaction (我们之前用过 Hmily 和公司架构组自研组件)。下面以电商下单流程为例对 TCC 做详细阐述。


流程图如下:

基本步骤如下:
  • 修改订单状态为“已支付”

  • 扣减库存

  • 扣减优惠券

  • 通知 WMS(仓储管理系统)捡货出库(异步消息)


我们先看扣减库存,更新订单状态和扣减优惠券这三步同步调用,通知 WMS 的异步消息会在后面的“基于消息的最终一致”部分详细阐述!

下面是伪代码,不同公司的产品逻辑会有差异,相关代码逻辑也可能会有不同,大家不用纠结代码逻辑正确性。

public void makePayment() {
   orderService.updateStatus(OrderStatus.Payed); //订单服务更新订单为已支付状态
   inventoryService.decrStock(); //库存服务扣减库存
   couponService.updateStatus(couponStatus.Used); //卡券服务更新优惠券为已使用状态      
}


看完这段代码,大家可能觉得很简单!那么有什么问题吗?答案是肯定的。没法保证数据一致性,也就是说不能保证这几步操作全部成功或者全部失败!


因为这几步操作是在分布式环境下进行的,每个操作分布在不同的服务中,不同的服务又对应不同的数据库,本地事务已经用不上了!


假如第一步更新订单为“已支付”成功了,第二步扣减库存时,库存服务挂了或者网络出问题了,导致扣减库存失败。你告诉用户支付成功了,但是库存没扣减。这怎么能行!


接下来,我们来看看TCC是如何帮我们解决这个问题并保证数据最终一致的。


TCC 分为两个阶段:
  • Try(预留冻结相关业务资源,设置临时状态,为下个阶段做准备)

  • Confirm 或者 Cancel(Confirm:对资源进行最终操作,Cancel:取消资源)


第一阶段 Try:
  • 更新订单状态:此时因为还没真正完成整个流程,订单状态不能直接改成已支付状态。可以加一个临时状态 Paying,表明订单正在支付中,支付结果暂时还不清楚!

  • 冻结库存:假设现在可销售库存 stock 是 10,这单扣减 1 个库存,别直接把库存减掉,而是在表中加一个冻结字段 locked_stock,locked_stock 加 1,再给 stock 减 1,这样就相当于冻结了 1 个库存。两个操作放在一个本地事务里。

  • 更新优惠券状态:优惠券加一个临时状态 Inuse,表明优惠券正在使用中,具体有没有正常被使用暂时还不清楚!


第二阶段 Confirm:假如第一阶段几个 Try 操作都成功了!既然第一阶段已经预留了库存,而且订单状态和优惠券状态也设置了临时状态,第二阶段的确认提交从业务上来说应该没什么问题了。


Confirm 阶段我们需要做下面三件事:
  • 先将订单状态从 Paying 改为已支付 Payed,订单状态也完成了。

  • 再将冻结的库存恢复 locked_stock 减 1,stock 第一阶段已经减掉 1 是 9 了,到此扣减库存就真正完成了。

  • 再将优惠券状态从 Inuse 改为 Used,表明优惠券已经被正常使用。


第二阶段 Cancel,假如第一阶段失败了:
  • 先将订单状态从 Paying 恢复为待支付 UnPayed。

  • 再将冻结的库存还回到可销售库存中,stock 加 1 恢复成 10,locked_stock 减 1,可以放在一个本地事务完成。

  • 再将优惠券状态从 Inuse 恢复为未使用 Unused。


基于 Hmily 框架的代码:

//订单服务
public class OrderService{

  //tcc接口
  @Hmily(confirmMethod = "confirmOrderStatus", cancelMethod = "cancelOrderStatus")
  public void makePayment() {
     更新订单状态为支付中
     冻结库存,rpc调用
     优惠券状态改为使用中,rpc调用
  }

  public void confirmOrderStatus() {
     更新订单状态为已支付
  }

  public void cancelOrderStatus() {
     恢复订单状态为待支付 
  }  

}


//库存服务
public class InventoryService {

  //tcc接口
  @Hmily(confirmMethod = "confirmDecr", cancelMethod = "cancelDecr")
  public void lockStock() {
     //防悬挂处理(下面有说明)
     if (分支事务记录表没有二阶段执行记录)
       冻结库存
     else
       return;
  }

  public void confirmDecr() {
     确认扣减库存
  }
  public void cancelDecr() {
     释放冻结的库存
  }  

}


//卡券服务
public class CouponService {

  //tcc接口
  @Hmily(confirmMethod = "confirm", cancelMethod = "cancel")
  public void handleCoupon() {
     //防悬挂处理(下面有说明)
     if (分支事务记录表没有二阶段执行记录)
       优惠券状态更新为临时状态Inuse
     else
       return;
  }

  public void confirm() {
     优惠券状态改为Used
  }
  public void cancel() {
     优惠券状态恢复为Unused
  }  

}


问题 1:有些朋友可能会问了,这些关于流程的逻辑也要手动编码吗?这也太麻烦了吧!


实际上 TCC 分布式事务框架帮我们把这些事都干了。比如我们前面提到的 Hmily,ByteTCC,TCC-transaction 这些框架。

因为 Try,Confirm,Cancel 这些操作都在 TCC 分布式事务框架控制范围之内,所以 Try 的各个步骤成功了或者失败了。


框架本身都知道,Try 成功了框架就会自动执行各个服务的 Confirm,Try 失败了框架就会执行各个服务的 Cancel(各个服务内部的 TCC 分布式事务框架会互相通信)。所以我们不用关心流程,只需要关注业务代码就可以啦!

问题 2:仔细想想,好像还有问题!假如 Confirm 阶段更新订单状态成功了,但是扣减库存失败了怎么办呢?

比如网络出问题了或者库存服务(宕机,重启)出问题了。当然,分布式事务框架也会考虑这些场景,框架会记录操作日志。


假如 Confirm 阶段扣减库存失败了,框架会不断重试调用库存服务直到成功(考虑性能问题,重试次数应该有限制)。Cancel 过程也是一样的道理。


注意,既然需要重试,我们就要保证接口的幂等性。什么?不太懂幂等性。简单说:一个操作不管请求多少次,结果都要保证一样。这里就不详细介绍啦!

再考虑一个场景,Try 阶段冻结库存的时候,因为是 RPC 远程调用,在网络拥塞等情况下,是有可能超时的。

假如冻结库存时发生超时,TCC 框架会回滚(Cancel)整个分布式事务,回滚完成后冻结库存请求才到达参与者(库存服务)并执行,这时被冻结的库存就没办法处理(恢复)了。


这种情况称之为“悬挂”,也就是说预留的资源后续无法处理。解决方案:第二阶段已经执行,第一阶段就不再执行了,可以加一个“分支事务记录表”,如果表里有相关第二阶段的执行记录,就不再执行 Try(上面代码有防悬挂处理)。


有人可能注意到还有些小纰漏,对,加锁,分布式环境下,我们可以考虑对第二阶段执行记录的查询和插入加上分布式锁,确保万无一失。

②基于消息的最终一致


还是以上面的电商下单流程为例:

上图,下单流程最后一步,通知 WMS 捡货出库,是异步消息走消息队列。

public void makePayment() {
   orderService.updateStatus(OrderStatus.Payed); //订单服务更新订单为已支付状态
   inventoryService.decrStock(); //库存服务扣减库存
   couponService.updateStatus(couponStatus.Used); //卡券服务更新优惠券为已使用状态      
   发送MQ消息捡货出库; //发送消息通知WMS捡货出库
}


按上面代码,大家不难发现问题!如果发送捡货出库消息失败,数据就会不一致!


有人说我可以在代码上加上重试逻辑和回退逻辑,发消息失败就重发,多次重试失败所有操作都回退。


这样一来逻辑就会特别复杂,回退失败要考虑,而且还有可能消息已经发送成功了,但是由于网络等问题发送方没得到 MQ 的响应,这些问题都要考虑进来!


幸好,有些消息队列帮我们解决了这些问题。比如阿里开源的 RocketMQ(目前已经是 Apache 开源项目),4.3.0 版本开始支持事务型消息(实际上早在贡献给 Apache 之前曾经支持过事务消息,后来被阉割了,4.3.0 版本重新开始支持事务型消息)。

先看看 RocketMQ 发送事务型消息的流程:
  • 发送半消息(所有事务型消息都要经历确认过程,从而确定最终提交或回滚(抛弃消息),未被确认的消息称为“半消息”或者“预备消息”,“待确认消息”)。

  • 半消息发送成功并响应给发送方。

  • 执行本地事务,根据本地事务执行结果,发送提交或回滚的确认消息。

  • 如果确认消息丢失(网络问题或者生产者故障等问题),MQ 向发送方回查执行结果。

  • 根据上一步骤回查结果,确定提交或者回滚(抛弃消息)。


看完事务型消息发送流程,有些读者可能没有完全理解,不要紧,我们来分析一下!


问题 1:假如发送方发送半消息失败怎么办?


半消息(待确认消息)是消息发送方发送的,如果失败,发送方自己是知道的并可以做相应处理。


问题 2:假如发送方执行完本地事务后,发送确认消息通知 MQ 提交或回滚消息时失败了(网络问题,发送方重启等情况),怎么办?


没关系,当 MQ 发现一个消息长时间处于半消息(待确认消息)的状态,MQ 会以定时任务的方式主动回查发送方并获取发送方执行结果。


这样即便出现网络问题或者发送方本身的问题(重启,宕机等),MQ 通过定时任务主动回查发送方基本都能确认消息最终要提交还是回滚(抛弃)。


当然出于性能和半消息堆积方面的考虑,MQ 本身也会有回查次数的限制。


问题 3:如何保证消费一定成功呢?


RocketMQ 本身有 Ack 机制,来保证消息能够被正常消费。如果消费失败(消息订阅方出错,宕机等原因),RocketMQ 会把消息重发回 Broker,在某个延迟时间点后(默认 10 秒后)重新投递消息。


结合上面几个同步调用 Hmily 完整代码如下:

//TransactionListener是rocketmq接口用于回调执行本地事务和状态回查
public class TransactionListenerImpl implements TransactionListener {
     //执行本地事务
     @Override
     public LocalTransactionState executeLocalTransaction(Message msg, Object arg) {
         记录orderID,消息状态键值对到共享map中,以备MQ回查消息状态使用;
         return LocalTransactionState.COMMIT_MESSAGE;
     }

     //回查发送者状态
     @Override
     public LocalTransactionState checkLocalTransaction(MessageExt msg) {
         String status = 从共享map中取出orderID对应的消息状态; 
         if("commit".equals(status))
           return LocalTransactionState.COMMIT_MESSAGE;
         else if("rollback".equals(status))
           return LocalTransactionState.ROLLBACK_MESSAGE;
         else
           return LocalTransactionState.UNKNOW;
     }
}


//订单服务
public class OrderService{

  //tcc接口
  @Hmily(confirmMethod = "confirmOrderStatus", cancelMethod = "cancelOrderStatus")
  public void makePayment() {
     1,更新订单状态为支付中
     2,冻结库存,rpc调用
     3,优惠券状态改为使用中,rpc调用
     4,发送半消息(待确认消息)通知WMS捡货出库 //创建producer时这册TransactionListenerImpl
  }

  public void confirmOrderStatus() {
     更新订单状态为已支付
  }

  public void cancelOrderStatus() {
     恢复订单状态为待支付 
  }  

}


//库存服务
public class InventoryService {

  //tcc接口
  @Hmily(confirmMethod = "confirmDecr", cancelMethod = "cancelDecr")
  public void lockStock() {
     //防悬挂处理
     if (分支事务记录表没有二阶段执行记录)
       冻结库存
     else
       return;
  }

  public void confirmDecr() {
     确认扣减库存
  }
  public void cancelDecr() {
     释放冻结的库存
  }  

}


//卡券服务
public class CouponService {

  //tcc接口
  @Hmily(confirmMethod = "confirm", cancelMethod = "cancel")
  public void handleCoupon() {
     //防悬挂处理
     if (分支事务记录表没有二阶段执行记录)
       优惠券状态更新为临时状态Inuse
     else
       return;
  }

  public void confirm() {
     优惠券状态改为Used
  }
  public void cancel() {
     优惠券状态恢复为Unused
  }  

}

如果执行到 TransactionListenerImpl.executeLocalTransaction 方法,说明半消息已经发送成功了。


也说明 OrderService.makePayment 方法的四个步骤都执行成功了,此时 TCC 也到了 Confirm 阶段。


所以在 TransactionListenerImpl.executeLocalTransaction 方法里可以直接返回 LocalTransactionState.COMMIT_MESSAGE 让 MQ 提交这条消息。


同时将该订单信息和对应的消息状态保存在共享 map 里,以备确认消息发送失败时 MQ 回查消息状态使用。


微服务化过程,无感知数据迁移


微服务化,其中一个重要意义在于数据隔离,即不同的服务对应各自的数据库表,避免不同业务模块间数据的耦合。


这也就意味着微服务化过程要拆分现有数据库,把单体数据库根据业务模块拆分成多个,进而涉及到数据迁移。


数据迁移过程我们要注意哪些关键点呢?
  • 第一,保证迁移后数据准确不丢失,即每条记录准确而且不丢失记录。

  • 第二,不影响用户体验(尤其是访问量高的C端业务需要不停机平滑迁移)。

  • 第三,保证迁移后的性能和稳定性。


数据迁移我们经常遇到的两个场景:
  • 业务重要程度一般或者是内部系统,数据结构不变,这种场景下可以采用挂从库,数据同步完找个访问低谷时间段,停止服务,然后将从库切成主库,再启动服务。简单省时,不过需要停服避免切主库过程数据丢失。

  • 重要业务,并发高,数据结构改变。这种场景一般需要不停机平滑迁移。下面就重点介绍这部分经历。


互联网行业,很多业务访问量很大,即便凌晨低谷时间,仍然有相当的访问量,为了不影响用户体验,很多公司对这些业务会采用不停机平滑迁移的方式。


因为对老数据迁移的同时,线上还不断有用户访问,不断有数据产生,不断有数据更新,所以我们不但要考虑老数据迁移的问题,还要考虑数据更新和产生新数据的问题。下面介绍一下我们之前的做法。


关键步骤如下:

①开启双写,新老库同时写入(涉及到代码改动)。注意:任何对数据库的增删改都要双写。


对于更新操作,如果新库没有相关记录,先从老库查出记录更新后写入数据库;为了保证写入性能,老库写完后,可以采用消息队列异步写入新库。


同时写两个库,不在一个本地事务,有可能出现数据不一致的情况,这样就需要一定的补偿机制来保证两个库数据最终一致。下一篇文章会分享最终一致性解决方案


②将某时间戳之前的老数据迁移到新库(需要脚本程序做老数据迁移,因为数据结构变化比较大的话,从数据库层面做数据迁移就很困难了)。


注意两点:

  • 时间戳一定要选择开启双写后的时间点,避免部分老数据被漏掉。

  • 迁移过程遇到记录冲突直接忽略(因为第一步有更新操作,直接把记录拉到了新库);迁移过程一定要记录日志,尤其是错误日志。


③第二步完成后,我们还需要通过脚本程序检验数据,看新库数据是否准确以及有没有漏掉的数据。


④数据校验没问题后,开启双读,起初新库给少部分流量,新老两库同时读取,由于时间延时问题,新老库数据可能有些不一致,所以新库读不到需要再读一遍老库。


逐步将读流量切到新库,相当于灰度上线的过程。遇到问题可以及时把流量切回老库。


⑤读流量全部切到新库后,关闭老库写入(可以在代码里加上可热配开关),只写新库。


⑥迁移完成,后续可以去掉双写双读相关无用代码。


第二步的老数据迁移脚本程序和第三步的检验程序可以工具化,以后再做类似的数据迁移可以复用。

目前各云服务平台也提供数据迁移解决方案,大家有兴趣也可以了解一下!


全链路 APM 监控


在体会到微服务带来好处的同时,很多公司也会明显感受到微服务化后那些让人头疼的问题。


比如,服务化之后调用链路变长,排查性能问题可能要跨多个服务,定位问题更加困难。


服务变多,服务间调用关系错综复杂,以至于很多工程师不清楚服务间的依赖和调用关系,之后的系统维护过程也会更加艰巨。诸如此类的问题还很多!


这时就迫切需要一个工具帮我们解决这些问题,于是 APM 全链路监控工具就应运而生了。


有开源的 Pinpoint、Skywalking 等,也有收费的 Saas 服务听云、OneAPM 等。有些实力雄厚的公司也会自研 APM。


下面我们介绍一下如何利用开源 APM 工具 Pinpoint 应对上述问题。


拓扑图:
微服务化后,服务数量变多,服务间调用关系也变得更复杂,以至于很多工程师不清楚服务间的依赖和调用关系,给系统维护带来很多困难。


通过拓扑图我们可以清晰地看到服务与服务,服务与数据库,服务与缓存中间件的调用和依赖关系。对服务关系了如指掌之后,也可以避免服务间循依赖、循环调用的问题。


请求调用栈(Call Stack)监控:

微服务化后,服务变多,调用链路变长,跨多个服务排查问题会更加困难。上图是一个请求的调用栈,我们可以清晰看到一次请求调用了哪些服务和方法、各个环节的耗时以及发生在哪个服节点。


上图的请求耗时过长,根据监控(红框部分)我们可以看到时间主要消耗在数据库 SQL 语句上。


点击数据库图表还可以看详细 SQL 语句,如下图:

如果发生错误,会显示为红色,错误原因也会直接显示出来。如下图:

类似性能问题和错误的线上排查。我们如果通过查日志的传统办法,可能会耗费大量的时间。但是通过 APM 工具分分钟就可以搞定了!


请求 Server Map:

Server Map 是 Pinpoint 另一个比较重要的功能。如上图,我们不但能清晰地看到一个请求的访问链路,而且还能看到每个节点的访问次数,为系统优化提供了有力的依据。


如果一次请求访问了多次数据库,说明代码逻辑可能有必要优化了!


JVM 监控:

此外,Pinpoint 还支持堆内存,活跃线程,CPU,文件描述符等监控。


关于微服务化,我们就分享这么多。希望对大家有帮助。

作者:二马读书

简介:曾任职于阿里巴巴,每日优鲜等互联网公司,任技术总监,15 年电商互联网经历。

编辑:陶家龙

出处:转载自微信公众号架构师进阶之路(ID:ermadushu)

精彩文章推荐:

我终于搞懂了微服务,太不容易了...
面试问Redis集群,被虐的不行了......
我为什么用ES做Redis监控,不用Prometheus或Zabbix?

关注公众号:拾黑(shiheibook)了解更多

[广告]赞助链接:

四季很好,只要有你,文娱排行榜:https://www.yaopaiming.com/
让资讯触达的更精准有趣:https://www.0xu.cn/

公众号 关注网络尖刀微信公众号
随时掌握互联网精彩
赞助链接