分布式鎖實(shí)例教程之防止重復(fù)提交
拋出一個(gè)問(wèn)題
需求:現(xiàn)在有一個(gè)常見(jiàn)的場(chǎng)景——用戶(hù)注冊(cè),但是如果出現(xiàn)重復(fù)提交的情況,則會(huì)出現(xiàn)多條注冊(cè)數(shù)據(jù),因此這里如何做好防止重復(fù)提交這是我們需要解決的問(wèn)題。
正常的代碼邏輯
1、注冊(cè)controller
/**
* 用戶(hù)注冊(cè)請(qǐng)求
* @param userDto
* @param bindingResult
* @return
*/
@RequestMapping(value=prefix+"/db/register",method = RequestMethod.POST,consumes = MediaType.APPLICATION_JSON_UTF8_VALUE)
public BaseResponse register(@RequestBody @Validated UserDto userDto, BindingResult bindingResult){
BaseResponse response=new BaseResponse(StatusCode.Success);
try {
log.debug("注冊(cè)信息: {} ",userDto);
//注冊(cè)之前,我們先判斷是否已經(jīng)注冊(cè)了。(正常邏輯)
User user=userService.selectByUserName(userDto.getUserName());
if (user!=null){
return new BaseResponse(StatusCode.UserNameExist);
}
userService.register(userDto);
}catch (Exception e){
e.printStackTrace();
response=new BaseResponse(StatusCode.Fail);
}
return response;
}
在controller中判斷用戶(hù)是否已經(jīng)注冊(cè),如果沒(méi)有注冊(cè),則調(diào)用注冊(cè)邏輯。
2、注冊(cè)service
/**
* 用戶(hù)注冊(cè)——最普通的操作,沒(méi)有任何加鎖,沒(méi)有任何防止重復(fù)提交
*
* @param userDto
* @return
* @throws Exception
*/
public int register(UserDto userDto) throws Exception {
int result = 0;
User user = new User();
BeanUtils.copyProperties(userDto, user);
result = userMapper.insertSelective(user);
return result;
}
簡(jiǎn)單的增加一個(gè)用戶(hù)信息。
問(wèn)題也很明顯,這樣畢竟會(huì)出現(xiàn)問(wèn)題,并發(fā)的問(wèn)題,也會(huì)出現(xiàn)重復(fù)注冊(cè)的情況。測(cè)試結(jié)果也很明顯

一堆重復(fù)注冊(cè)的,但是加入分布式鎖就好了么?
3、加入分布式鎖,問(wèn)題依舊
分布式鎖的實(shí)現(xiàn)方式
/**
* 用戶(hù)注冊(cè),基于redisson的分布式鎖
*
* @param userDto
* @return
*/
public int registerLockRedisson(UserDto userDto) {
int result = 0;
RLock rLock = redissonLockComponent.acquireLock(userDto.getUserName());
try {
if (rLock != null) {
User user = new User();
BeanUtils.copyProperties(userDto, user);
user.setCreateTime(new Date());
userMapper.insertSelective(user);
}
} catch (Exception e) {
log.error("獲取redisson分布式鎖異常");
} finally {
if (rLock != null) {
redissonLockComponent.releaseLock(rLock);
}
}
return result;
}
加入分布式鎖之后,再進(jìn)行測(cè)試。

不好意思,依舊出現(xiàn)了重復(fù)注冊(cè)的情況。何解?
問(wèn)題分析,為了遵循單一職責(zé),這里的讀取數(shù)據(jù)(判斷是否注冊(cè))與寫(xiě)入數(shù)據(jù)(用戶(hù)注冊(cè))操作是分開(kāi)的,分布式鎖為了進(jìn)一步細(xì)化,只是加在了寫(xiě)入數(shù)據(jù)階段,并沒(méi)有加在整個(gè)業(yè)務(wù)階段,因此會(huì)出現(xiàn)數(shù)據(jù)重復(fù)提交的問(wèn)題,解決方法有很多,最暴力的方法無(wú)非就是給數(shù)據(jù)庫(kù)user表中的用戶(hù)名字段加入唯一約束。但是這樣隨著業(yè)務(wù)規(guī)模擴(kuò)大,數(shù)據(jù)庫(kù)壓力會(huì)越來(lái)越大。
解決方法
解決方法有幾種,前面提到的給數(shù)據(jù)庫(kù)增加唯一索引也是一種方法。但是為了減輕數(shù)據(jù)庫(kù)的壓力,這種操作可以直接在應(yīng)用層處理。
分布式鎖+防重操作
在分布式鎖的基礎(chǔ)上,加入redis存儲(chǔ)key值,作為防重提交的判斷。不想過(guò)多解釋了,直接上代碼吧。
/**
* 用戶(hù)注冊(cè),redisson分布式鎖,redis防止重復(fù)提交
*
* @param userDto
* @return
*/
public int registerLockAvoidDupPost(UserDto userDto) {
int result = 0;
RLock rLock = redissonLockComponent.acquireLock(userDto.getUserName());
try {
//redis中根據(jù)用戶(hù)名存儲(chǔ)作為key值
String key = lockKeyPrefix+userDto.getUserName();
if (!stringRedisTemplate.hasKey(key)) {//如果不存在key則進(jìn)入注冊(cè)階段
stringRedisTemplate.opsForValue().set(key,UUID.randomUUID().toString(),10L,TimeUnit.SECONDS);
User user = new User();
BeanUtils.copyProperties(userDto, user);
user.setCreateTime(new Date());
userMapper.insertSelective(user);
log.info("{},注冊(cè)成功",userDto.getUserName());
}else{//如果存在,則提示不可重復(fù)提交
log.error("10秒內(nèi),請(qǐng)勿重復(fù)提交注冊(cè)信息");
}
} catch (Exception e) {
log.error("獲取redisson分布式鎖異常");
} finally {
if (rLock != null) {
redissonLockComponent.releaseLock(rLock);
}
}
return result;
}
分布式鎖的實(shí)現(xiàn)方式有多重,redis/redisson/zookeeper等,只需要在已經(jīng)實(shí)現(xiàn)分布式鎖的基礎(chǔ)上引入防重提交的機(jī)制即可。
因此還有其他方式的實(shí)現(xiàn),如下所示為zookeeper分布式鎖+redis防重的方式
/**
* 用戶(hù)注冊(cè),redisson分布式鎖,redis防止重復(fù)提交
*
* @param userDto
* @return
*/
public int registerLockAvoidDupPost(UserDto userDto) {
int result = 0;
InterProcessMutex mutex=new InterProcessMutex(client,zkPrefix+userDto.getUserName()+"-lock");
try {
if (mutex.acquire(10L, TimeUnit.SECONDS)){
final String realKey=zkRedisKeyPrefix+userDto.getUserName();
if (!stringRedisTemplate.hasKey(realKey)){
stringRedisTemplate.opsForValue().set(realKey, UUID.randomUUID().toString());
User user=new User();
BeanUtils.copyProperties(userDto,user);
user.setCreateTime(new Date());
userMapper.insertSelective(user);
log.info("{},注冊(cè)成功",userDto.getUserName());
}else{
log.error("10秒內(nèi),請(qǐng)勿重復(fù)提交注冊(cè)信息");
}
}else{
throw new RuntimeException("獲取zk分布式鎖失敗!");
}
}catch (Exception e){
e.printStackTrace();
throw e;
}finally {
mutex.release();
}
return result;
}
測(cè)試結(jié)果:

并不會(huì)出現(xiàn)重復(fù)注冊(cè)情況了。
總結(jié)
防重提交不能全部交給數(shù)據(jù)庫(kù)
到此這篇關(guān)于分布式鎖實(shí)例教程之防止重復(fù)提交的文章就介紹到這了,更多相關(guān)分布式鎖防止重復(fù)提交內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Spring注解@Configuration與@Bean注冊(cè)組件的使用詳解
這篇文章主要介紹了SpringBoot中的注解@Configuration與@Bean注冊(cè)組件的使用,具有很好的參考價(jià)值,希望對(duì)大家有所幫助2022-06-06
solr 配置中文分析器/定義業(yè)務(wù)域/配置DataImport功能方法(測(cè)試用)
下面小編就為大家?guī)?lái)一篇solr 配置中文分析器/定義業(yè)務(wù)域/配置DataImport功能方法(測(cè)試用)。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-09-09
解決springboot環(huán)境切換失效的問(wèn)題
這篇文章主要介紹了解決springboot環(huán)境切換失效的問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-09-09
Java設(shè)計(jì)模式之Builder建造者模式
這篇文章主要為大家詳細(xì)介紹了Java設(shè)計(jì)模式之Builder建造者模式的相關(guān)資料,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-03-03
jpa異常No entity found for query問(wèn)題解決
這篇文章主要為大家介紹了jpa異常之No entity found for query的異常問(wèn)題解決,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步2022-03-03
java String類(lèi)型對(duì)象轉(zhuǎn)換為自定義類(lèi)型對(duì)象的實(shí)現(xiàn)
本文主要介紹了java String類(lèi)型對(duì)象轉(zhuǎn)換為自定義類(lèi)型對(duì)象的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2023-06-06
關(guān)于Spring @Bean 相同加載順序不同結(jié)果不同的問(wèn)題記錄
本文主要探討了在Spring 5.1.3.RELEASE版本下,當(dāng)有兩個(gè)全注解類(lèi)定義相同類(lèi)型的Bean時(shí),由于加載順序不同,最終生成的Bean實(shí)例也會(huì)不同,文章通過(guò)分析ConfigurationClassPostProcessor的執(zhí)行過(guò)程,解釋了BeanDefinition的加載和覆蓋機(jī)制,感興趣的朋友一起看看吧2025-02-02

