返回顶部
首页 > 资讯 > 后端开发 > GO >Golang怎么实现可重入锁
  • 487
分享到

Golang怎么实现可重入锁

2023-06-30 16:06:37 487人浏览 薄情痞子
摘要

本文小编为大家详细介绍“golang怎么实现可重入锁”,内容详细,步骤清晰,细节处理妥当,希望这篇“Golang怎么实现可重入锁”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。什么是可重入锁我们平时说的分布式锁,一

本文小编为大家详细介绍“golang怎么实现可重入”,内容详细,步骤清晰,细节处理妥当,希望这篇“Golang怎么实现可重入锁”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。

什么是可重入锁

我们平时说的分布式锁,一般指的是在不同服务器上的多个线程中,只有一个线程能抢到一个锁,从而执行一个任务。而我们使用锁就是保证一个任务只能由一个线程来完成。所以我们一般是使用这样的三段式逻辑:

Lock();
DoJob();
Unlock();

但是由于我们的系统都是分布式的,这个锁一般不会只放在某个进程中,我们会借用第三方存储,比如 Redis 来做这种分布式锁。但是一旦借助了第三方存储,我们就必须面对这个问题:Unlock是否能保证一定运行呢?

这个问题,我们面对的除了程序的bug之外,还有网络的不稳定,进程被杀死,服务器被down机等。我们是无法保证Unlock一定被运行的。

那么我们就一般在Lock的时候为这个锁加一个超时时间作为兜底。

LockByExpire(duration);
DoJob();
Unlock();

这个超时时间是为了一旦出现异常情况导致Unlock没有被运行,这个锁在duration时间内也会被自动释放。这个在Redis中我们一般就是使用set ex 来进行锁超时的设定。

但是有这个超时时间我们又遇上了问题,超时时间设置多久合适呢?当然要设置的比 DoJob 消耗的时间更长,否则的话,在任务还没结束的时候,锁就被释放了,还是有可能导致并发任务的存在。

但是实际上,同样由于网络超时问题,系统运行状况问题等,我们是无法准确知道DoJob这个函数要执行多久的。那么这时候怎么办呢?

有两个办法:

第一个方法,我们可以对DoJob做一个超时设置。让DoJob最多只能执行n秒,那么我的分布式锁的超时时长设置比n秒长就可以了。为一个任务设置超时时间在很多语言是可以做到的。比如golang 中的 TimeoutContext。

而第二种方法,就是我们先为锁设置一个比较小的超时时长,然后不断续期这个锁。对一个锁的不断需求,也可以理解为重新开始加锁,这种可以不断续期的锁,就叫做可重入锁。

除了主线程之外,可重入锁必然有一个另外的线程(或者携程)可以对这个锁进行续期,我们叫这个额外的程序叫做watchDog(看门狗)。

具体实现

在Golang中,语言级别天生支持协程,所以这种可重入锁就非常容易实现:

// DistributeLockRedis 基于redis的分布式可重入锁,自动续租type DistributeLockRedis struct {key       string             // 锁的keyexpire    int64              // 锁超时时间status    bool               // 上锁成功标识cancelFun context.CancelFunc // 用于取消自动续租携程redis     redis.Client       // redis句柄}// 创建可func NewDistributeLockRedis(key string, expire int64) *DistributeLockRedis {return &DistributeLockRedis{ key : key, expire : expire,}}// TryLock 上锁func (dl *DistributeLockRedis) TryLock() (err error) {if err = dl.lock(); err != nil {return err}ctx, cancelFun := context.WithCancel(context.Background())dl.cancelFun = cancelFundl.startWatchDog(ctx) // 创建守护协程,自动对锁进行续期dl.status = truereturn nil}// competition 竞争锁func (dl *DistributeLockRedis) lock() error {if res, err := redis.String(dl.redis.Do(context.Background(), "SET", dl.key, 1, "NX", "EX", dl.expire)); err != nil {return err} return nil}// guard 创建守护协程,自动续期func (dl *DistributeLockRedis) startWatchDog(ctx context.Context) {safeGo(func() error {for {select {// Unlock通知结束case <-ctx.Done():return nildefault:// 否则只要开始了,就自动重入(续租锁)if dl.status {if res, err := redis.Int(dl.redis.Do(context.Background(), "EXPIRE", dl.key, dl.expire)); err != nil {return nil} // 续租时间为 expire/2 秒time.Sleep(time.Duration(dl.expire/2) * time.Second)}}}})}// Unlock 释放锁func (dl *DistributeLockRedis) Unlock() (err error) {// 这个重入锁必须取消,放在第一个地方执行if dl.cancelFun != nil {dl.cancelFun() // 释放成功,取消重入锁}var res intif dl.status {if res, err = redis.Int(dl.redis.Do(context.Background(), "Del", dl.key)); err != nil {return fmt.Errorf("释放锁失败")}if res == 1 {dl.status = falsereturn nil}}return fmt.Errorf("释放锁失败")}

这段代码的逻辑基本上都以注释的形式来写了。其中主要就在startWatchDog,对锁进行重新续期

ctx, cancelFun := context.WithCancel(context.Background())dl.cancelFun = cancelFundl.startWatchDog(ctx) // 创建守护协程,自动对锁进行续期dl.status = true

首先创建一个cancelContext,它的context函数cancelFunc是给Unlock进行调用的。然后启动一个goroutine进程来循环续期。

这个新启动的goroutine在主goroutine处理结束,调用Unlock的时候,才会结束,否则会在 过期时间/2 的时候,调用一次redis的expire命令来进行续期。

至于外部,在使用的时候如下

func Foo() error {  key := foo    // 创建可重入的分布式锁dl := NewDistributeLockRedis(key, 10)// 争抢锁err := dl.TryLock()if err != nil {// 没有抢到锁return err}// 抢到锁的记得释放锁defer func() {dl.Unlock()}// 做真正的任务DoJob()}

读到这里,这篇“Golang怎么实现可重入锁”文章已经介绍完毕,想要掌握这篇文章的知识点还需要大家自己动手实践使用过才能领会,如果想了解更多相关内容的文章,欢迎关注编程网GO频道。

您可能感兴趣的文档:

--结束END--

本文标题: Golang怎么实现可重入锁

本文链接: https://lsjlt.com/news/330111.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • Golang怎么实现可重入锁
    本文小编为大家详细介绍“Golang怎么实现可重入锁”,内容详细,步骤清晰,细节处理妥当,希望这篇“Golang怎么实现可重入锁”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。什么是可重入锁我们平时说的分布式锁,一...
    99+
    2023-06-30
  • Golang实现可重入锁的示例代码
    目录什么是可重入锁具体实现项目中遇到了可重入锁的需求和实现,具体记录下。 什么是可重入锁 我们平时说的分布式锁,一般指的是在不同服务器上的多个线程中,只有一个线程能抢到一个锁,从而执...
    99+
    2024-04-02
  • 怎么实现Java可重入分布式锁
    本篇内容主要讲解“怎么实现Java可重入分布式锁”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“怎么实现Java可重入分布式锁”吧!可重入说到可重入锁,首先我们来看看一段来自 wiki 上可重入的...
    99+
    2023-06-16
  • java中怎么实现可重入的自旋锁
    这篇文章主要介绍了java中怎么实现可重入的自旋锁的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇java中怎么实现可重入的自旋锁文章都会有所收获,下面我们一起来看看吧。说明是指试图获得锁的线程不会堵塞,而是通过...
    99+
    2023-06-30
  • 怎么在java中实现内置锁的可重入性
    这篇文章给大家介绍怎么在java中实现内置锁的可重入性,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。java基本数据类型有哪些Java的基本数据类型分为:1、整数类型,用来表示整数的数据类型。2、浮点类型,用来表示小数...
    99+
    2023-06-14
  • Golang中怎么实现一个不可重入函数
    Golang中怎么实现一个不可重入函数,相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。场景用例某个服务是对某些条件进行轮询,每秒监视一些状态。我们希望每个状态都可以独立地检查,而...
    99+
    2023-06-04
  • 怎么在java中实现一个可重入的自旋锁
    怎么在java中实现一个可重入的自旋锁?相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。Java可以用来干什么Java主要应用于:1. web开发;2. Android开发;3. ...
    99+
    2023-06-14
  • redis分布式锁之可重入锁的实现代码
    上篇redis实现的分布式锁,有一个问题,它不可重入。 所谓不可重入锁,即若当前线程执行某个方法已经获取了该锁,那么在方法中尝试再次获取锁时,就会获取不到被阻塞。 同一个人拿一个锁 ...
    99+
    2024-04-02
  • Redis如何实现可重入锁的设计
    这篇文章主要介绍Redis如何实现可重入锁的设计,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!但是仍然有些场景是不满⾜的,例如⼀ 个⽅法获取到锁之后,可能在⽅法内调这个⽅法此时就获取...
    99+
    2024-04-02
  • Curator实现分布式锁(可重入 不可重入 读写 联锁 信号量 栅栏 计数器)
    文章目录 前言代码实践1. 配置2. 可重入锁InterProcessMutex3. 不可重入锁InterProcessSemaphoreMutex4. 可重入读写锁InterProcessReadWriteLock5. 联锁Int...
    99+
    2023-08-19
    分布式 java 服务器
  • Redisson可重入锁解锁逻辑是什么
    这篇文章主要介绍“Redisson可重入锁解锁逻辑是什么”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“Redisson可重入锁解锁逻辑是什么”文章能帮助大家解决问题。相比较Redisson可重入锁的...
    99+
    2023-07-05
  • 使用python实现可重入的公平读写锁
    在本项目中,读写锁主要应用于多线程服务器场景下的日志文件的读写,以及缓存的获取和更新。 多线程编程的准标准库posix pthread库拥有rwlock, 而python2.7自带的threading库没有读写锁,只有可重入锁RL...
    99+
    2023-01-31
    公平 python 可重入
  • golang自旋锁怎么实现
    Golang中的自旋锁可以通过sync包中的Mutex类型来实现。Mutex类型提供了两个方法:Lock()用于获取锁,Unlock...
    99+
    2023-10-26
    golang
  • 详解Java ReentrantLock可重入,可打断,锁超时的实现原理
    目录概述可重入可打断锁超时概述 前面讲解了ReentrantLock加锁和解锁的原理实现,但是没有阐述它的可重入、可打断以及超时获取锁失败的原理,本文就重点讲解这三种情况。建议大家先...
    99+
    2022-11-13
    Java ReentrantLock可重入 Java ReentrantLock可打断 Java ReentrantLock锁超时 Java ReentrantLock
  • 一篇文章让你彻底了解Java可重入锁和不可重入锁
    可重入锁  广义上的可重入锁指的是可重复可递归调用的锁,在外层使用锁之后,在内层仍然可以使用,并且不发生死锁(前提得是同一个对象或者class),这样的锁就叫做可重入锁。 ...
    99+
    2024-04-02
  • golang不可重入是什么意思
    在计算机科学领域中,重入是指一个函数在执行期间可以被多次调用的特性。但是,有些语言或环境并不支持函数的重入。在 Golang 中,函数不是可重入的,这是由于 Golang 本身的设计决策所导致的。首先,让我们了解什么是不可重入函数。一个不可...
    99+
    2023-05-14
  • Golang怎么实现死锁?怎么避免?
    Golang 是一门高效、强类型、静态类型的编程语言,由 Google 开发,目的是为了解决一些类似于死锁等问题。虽然 Golang 实现死锁十分困难,但是在这篇文章中,我们将要探讨如何使用 Golang 实现死锁。什么是死锁?死锁指的是多...
    99+
    2023-05-14
  • Redisson可重入锁解锁逻辑详细讲解
    目录主动释放自动释放本篇文章基于redisson-3.17.6版本源码进行分析 相比较Redisson可重入锁的加锁逻辑,释放锁的逻辑就相对简单一些。释放锁分为主动释放和自动释放两种...
    99+
    2023-02-11
    Redisson可重入锁 Redisson可重入锁解锁
  • Springboot基于Redisson实现Redis分布式可重入锁源码解析
    目录一、前言二、为什么使用Redisson1. 我们打开官网2. 我们可以看到官方让我们去使用其他3. 打开官方推荐4. 找到文档三、Springboot整合Redisson1. 导...
    99+
    2024-04-02
  • Java 重入锁和读写锁怎么使用
    这篇文章主要介绍“Java 重入锁和读写锁怎么使用”,在日常操作中,相信很多人在Java 重入锁和读写锁怎么使用问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Java 重入锁和读写锁怎么使用”的疑惑有所帮助!...
    99+
    2023-06-08
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作