锁主要解决资源并发的问题,在java单应用的情况下,可以通过Synchronized,ReentrantLock实现。但在分布式环境下对同一资源访问,如何实现并发锁控制呢?主要实现方案是分布式锁。本文主要介绍Redis分布式锁的一种设计方案,代码预计在第二篇发布。将传统的基于Jar包的锁提升为锁服务。不足之处,请大家一起交流。

一、整体方案

1.1组件图

 

分布式服务架构用什么框架 分布式服务设计_类图

           

1.2组件介绍

(1)锁服务-客户端

主要职责:封装LockService的调用(可使用代理或注解方式,用于业务功能使用)。

(2)锁服务-服务端

         主要职责:实现加锁和解锁功能(手动解锁,自动解锁)。

(3)锁服务-管理端

         主责职责:监控锁服务的运行情况,控制锁参数的执行细节。

2.3锁生命周期

锁的状态:无锁,已锁定,锁定成功,已解锁[无锁]

锁的操作:请求加锁,请求解锁,自动解锁,锁等待,锁超时

分布式服务架构用什么框架 分布式服务设计_分布式服务架构用什么框架_02

二、锁服务-服务端设计

2.1设计类图

分布式服务架构用什么框架 分布式服务设计_分布式服务架构用什么框架_03

2.2类图介绍

(1)对外开放锁定和解锁接口,其中Key的参数为Map<String,String>类型。

(2)锁机制基于Redis的串行执行和SetNx命令实现。

(3)lock(keyMap : Map<String, String>)方法,采用默认超时时间,此时间可以通过管理端根据不同的业务参数自定义。

(4)lock(keyMap : Map<String, String>, ltimeout :long) ,采用指定超时时间,此时间也可以通过管理端根据不同的业务参数自定义覆盖或不覆盖。

2.3 Map Key参数设计

基于Map实现可扩展的参数设计,主要字段介绍,如下:

Map key

Map value

描述

格式说明

source

100

业务系统


ver

10

版本号


subSys

101010

子系统

前两位=子系统,中间两位=模块,后两位=功能

业务参数N


其它业务参数

N个则为N个key

三、锁服务-客户端设计

3.1类图设计

 

分布式服务架构用什么框架 分布式服务设计_分布式服务架构用什么框架_04

3.2类图介绍

(1)左侧是调用锁服务的简单封装,主要用于封装重复代码,方便业务调用。简称为代理模式;

(2)右侧是在代理模式的基础上,做注解和拦截,根据复杂程度此处可实现可不实现;

(3)如果简单,代理模式也可以不使用,直接调用服务接口即可。

四、锁服务-管理端设计

4.1功能用例图

分布式服务架构用什么框架 分布式服务设计_代理模式_05

4.2主要用例介绍

(1)开通锁权限

开通某业务系统的分布式锁使用权限,可细化到业务系统,模块或功能。

(2)管理锁权限

         管理锁的使用权限比如停用,开通,查询等功能。

(3)配置通用锁参数

         配置锁服务的公用参数,比如默认锁超时时间,服务降级,异常返回值等。

(4)配置业务锁参数

         配置具体业务,模块,功能,参数的锁参数,比如默认超时时间,服务降级,异常返回值等。

(5)查看锁运行情况

根据实际需要可开发,如查看目前锁定的业务功能,锁定时间,锁冲突日志,历史锁定统计等功能。

 

五、锁分级-实际应用

5.1分级锁设计

针对不同的业务场景,锁控制的范围和粒度,简单划分为以下几种锁类型。

业务锁:与锁定值相关的所有操作;比如:锁订单号,可以控制所有与订单号有关的流程;

业务锁:一个业务流程中的某一类数据;比如在落单时控制并发,Key=订单号+_CREATE。

任务锁:锁定整个方法;比如定时任务中的锁;key=SendEmailTask

乐观锁:允许并发操作,但最终只有一个可执行成功,其它失败。适用于并发量不大但需要强一致的情况。比如将某订单从待支付更新为支付成功。

5.2锁服务降级

         当分布式锁宕机时,需要支持锁服务降级,此处推荐两种方案:

(1)数据库方式

原理:数据库唯一索引+DB乐观锁[最底层控制方案]+查询数据。适用于并发量不大的情况,如果并发量大需要前端做限流。

(2)基于Zookeeper

         原理:在获取分布式锁的时候在locker节点下创建临时顺序节点,释放锁的时候删除该临时节点。适合大并发场景。此方案也可以理解为与Redis锁的互备,实现高可用。

六、下篇预告

设计方案优化,服务化实现,提供源码。