热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

实现一个redis的分布式锁

分布式锁的实现有挺多细节要注意。1.要设置过期时间,避免释放锁的时候失败了,锁长期得不到释放导致的死锁问题2.要设置锁的拥有者请求一拿到锁,开始执行业务,业务执行时长超过锁设置的过






分布式锁的实现有挺多细节要注意。
1.要设置过期时间,避免释放锁的时候失败了,锁长期得不到释放导致的死锁问题
2.要设置锁的拥有者


请求一拿到锁,开始执行业务,业务执行时长超过锁设置的过期时间时,锁过期了,假设这个时候请求二拿到锁,刚开始执行业务,请求一业务执行完成,开始释放锁。因为没有设置锁的拥有者,导致请求一释放了请求二的锁,就会出现问题。


具体代码:

package redislock
import (
"context"
"time"
"github.com/go-redis/redis/v8"
)
//Lock is a struct that handle config and context
type Lock struct {
Config *Config
context context.Context
}
//Config is a struct that maintains redis client
type Config struct {
Client *redis.Client
}
//New is a method that return a instance of Lock
func New(c *Config) *Lock {
return &Lock{
Config: c,
context: context.Background(),
}
}
//Get is a method that try to get distribution lock
func (l *Lock) Get(key string, ttl time.Duration, owner string) (bool, error) {
return l.Config.Client.SetNX(l.context, key, owner, ttl).Result()
}
//Release is a method that release lock
func (l *Lock) Release(key string, owner string) (bool, error) {
luaScript := `
if redis.call("get",KEYS[1]) == ARGV[1] then
return redis.call("del",KEYS[1])
else
return 0
end
`

res, err := l.Config.Client.Eval(l.context, luaScript, []string{key}, owner).Result()
if res.(int64) == 0 {
return false, err
}
return true, nil
}

但仅仅是这样是不够的,因为Get方法只试了一次,并没有实现锁的自旋,我们应该写一个LoopGet方法去循环尝试获取锁。

//LoopGet is a method that try to get distribution lock looply
func (l *Lock) LoopGet(key string, ttl time.Duration, owner string) (chan bool, error) {
c := make(chan bool, 1)
for {
if res, err := l.Get(key, ttl, owner); res {
if err != nil {
c <- false
return c, err
}
c <- res
break
}
}
go func() {
defer close(c)
for {
if len(c) == 0 {
break
}
time.Sleep(time.Millisecond * 800)
}
}()
return c, nil
}

不停地尝试获取锁,成功之后返回channel,记得开一个协程回收channel,当channel的缓冲数据被读取后,就回收该channel,避免内存泄漏。




go


推荐阅读
  • 标题: ... [详细]
  • 本文介绍了在处理不规则数据时如何使用Python自动提取文本中的时间日期,包括使用dateutil.parser模块统一日期字符串格式和使用datefinder模块提取日期。同时,还介绍了一段使用正则表达式的代码,可以支持中文日期和一些特殊的时间识别,例如'2012年12月12日'、'3小时前'、'在2012/12/13哈哈'等。 ... [详细]
  • 本文介绍了RxJava在Android开发中的广泛应用以及其在事件总线(Event Bus)实现中的使用方法。RxJava是一种基于观察者模式的异步java库,可以提高开发效率、降低维护成本。通过RxJava,开发者可以实现事件的异步处理和链式操作。对于已经具备RxJava基础的开发者来说,本文将详细介绍如何利用RxJava实现事件总线,并提供了使用建议。 ... [详细]
  • Spring源码解密之默认标签的解析方式分析
    本文分析了Spring源码解密中默认标签的解析方式。通过对命名空间的判断,区分默认命名空间和自定义命名空间,并采用不同的解析方式。其中,bean标签的解析最为复杂和重要。 ... [详细]
  • Linux重启网络命令实例及关机和重启示例教程
    本文介绍了Linux系统中重启网络命令的实例,以及使用不同方式关机和重启系统的示例教程。包括使用图形界面和控制台访问系统的方法,以及使用shutdown命令进行系统关机和重启的句法和用法。 ... [详细]
  • 本文介绍了Redis的基础数据结构string的应用场景,并以面试的形式进行问答讲解,帮助读者更好地理解和应用Redis。同时,描述了一位面试者的心理状态和面试官的行为。 ... [详细]
  • Java容器中的compareto方法排序原理解析
    本文从源码解析Java容器中的compareto方法的排序原理,讲解了在使用数组存储数据时的限制以及存储效率的问题。同时提到了Redis的五大数据结构和list、set等知识点,回忆了作者大学时代的Java学习经历。文章以作者做的思维导图作为目录,展示了整个讲解过程。 ... [详细]
  • 本文讨论了一个关于cuowu类的问题,作者在使用cuowu类时遇到了错误提示和使用AdjustmentListener的问题。文章提供了16个解决方案,并给出了两个可能导致错误的原因。 ... [详细]
  • sklearn数据集库中的常用数据集类型介绍
    本文介绍了sklearn数据集库中常用的数据集类型,包括玩具数据集和样本生成器。其中详细介绍了波士顿房价数据集,包含了波士顿506处房屋的13种不同特征以及房屋价格,适用于回归任务。 ... [详细]
  • 网络请求模块选择——axios框架的基本使用和封装
    本文介绍了选择网络请求模块axios的原因,以及axios框架的基本使用和封装方法。包括发送并发请求的演示,全局配置的设置,创建axios实例的方法,拦截器的使用,以及如何封装和请求响应劫持等内容。 ... [详细]
  • 本文介绍了在iOS开发中使用UITextField实现字符限制的方法,包括利用代理方法和使用BNTextField-Limit库的实现策略。通过这些方法,开发者可以方便地限制UITextField的字符个数和输入规则。 ... [详细]
  • MPLS VP恩 后门链路shamlink实验及配置步骤
    本文介绍了MPLS VP恩 后门链路shamlink的实验步骤及配置过程,包括拓扑、CE1、PE1、P1、P2、PE2和CE2的配置。详细讲解了shamlink实验的目的和操作步骤,帮助读者理解和实践该技术。 ... [详细]
  • 本文介绍了在MFC下利用C++和MFC的特性动态创建窗口的方法,包括继承现有的MFC类并加以改造、插入工具栏和状态栏对象的声明等。同时还提到了窗口销毁的处理方法。本文详细介绍了实现方法并给出了相关注意事项。 ... [详细]
  • 本文介绍了Java中Currency类的getInstance()方法,该方法用于检索给定货币代码的该货币的实例。文章详细解释了方法的语法、参数、返回值和异常,并提供了一个示例程序来说明该方法的工作原理。 ... [详细]
  • 用Vue实现的Demo商品管理效果图及实现代码
    本文介绍了一个使用Vue实现的Demo商品管理的效果图及实现代码。 ... [详细]
author-avatar
笑看斜阳
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有