如何用go開(kāi)發(fā)基于redis的分布式鎖拧略。

什么是分布式鎖

分布式鎖是用于解決分布式系統(tǒng)控制協(xié)調(diào)任務(wù)用的,保證分布式服務(wù)中寒锚,能達(dá)到控制任務(wù)的目的。

常見(jiàn)的分布式鎖

  • 基于zookeeper
  • 基于etcd
  • 基于consul
  • 基于redis

為啥用redis實(shí)現(xiàn)分布式鎖

首先因?yàn)楹?jiǎn)單违孝,redis中有個(gè)命令是setnx, 這個(gè)命令就能起到分布式鎖的作用刹前,已經(jīng)存在key就會(huì)設(shè)置失敗。
其次redis是一般項(xiàng)目中都有的組件雌桑,如果采用其他的喇喉,可能會(huì)增加成本。

用go開(kāi)發(fā)基于redis的分布式鎖

大致功能如下

  • 設(shè)置過(guò)期時(shí)間
    防止單點(diǎn)服務(wù)異常校坑,鎖不釋放
  • 自動(dòng)續(xù)租
    防止鎖超時(shí)問(wèn)題拣技,如設(shè)置過(guò)期時(shí)間2秒,但是中間的任務(wù)如執(zhí)行幾條sql語(yǔ)句用了5秒耍目,那么這個(gè)鎖就可能是無(wú)效了膏斤,很可能被其他服務(wù)給占用,導(dǎo)致sql任務(wù)的異常出現(xiàn)邪驮。 當(dāng)鎖住2/3過(guò)期時(shí)間還未過(guò)期的莫辨,我們可以對(duì)他續(xù)租。 這樣鎖就還是有效的耕捞。
  • redisClient可以使用全局的衔掸,也可以有用戶傳入

首先我們用到最常見(jiàn)的go-redis庫(kù)

go get github.com/go-redis/redis/v8

核心代碼就200行烫幕。

全局redisClient實(shí)現(xiàn)

package goredislock

import (
    "context"
    "fmt"
    "log"
    "sync"
    "time"

    "github.com/go-redis/redis/v8"
)

var GlobalRedisClient *redis.Client
var redisClientOnce sync.Once

func NewRedisClient(addr string) *redis.Client {
    redisClientOnce.Do(func() {
        GlobalRedisClient = redis.NewClient(&redis.Options{
            Network:  "tcp",
            Addr:     addr,
            Password: "", //密碼
            DB:       0,  // redis數(shù)據(jù)庫(kù)

            //連接池容量及閑置連接數(shù)量
            PoolSize:     15, // 連接池?cái)?shù)量
            MinIdleConns: 10, //好比最小連接數(shù)
            //超時(shí)
            DialTimeout:  5 * time.Second, //連接建立超時(shí)時(shí)間
            ReadTimeout:  3 * time.Second, //讀超時(shí)俺抽,默認(rèn)3秒, -1表示取消讀超時(shí)
            WriteTimeout: 3 * time.Second, //寫(xiě)超時(shí)较曼,默認(rèn)等于讀超時(shí)
            PoolTimeout:  4 * time.Second, //當(dāng)所有連接都處在繁忙狀態(tài)時(shí)磷斧,客戶端等待可用連接的最大等待時(shí)長(zhǎng),默認(rèn)為讀超時(shí)+1秒。

            //閑置連接檢查包括IdleTimeout弛饭,MaxConnAge
            IdleCheckFrequency: 60 * time.Second, //閑置連接檢查的周期冕末,默認(rèn)為1分鐘,-1表示不做周期性檢查侣颂,只在客戶端獲取連接時(shí)對(duì)閑置連接進(jìn)行處理档桃。
            IdleTimeout:        5 * time.Minute,  //閑置超時(shí)
            MaxConnAge:         0 * time.Second,  //連接存活時(shí)長(zhǎng),從創(chuàng)建開(kāi)始計(jì)時(shí)憔晒,超過(guò)指定時(shí)長(zhǎng)則關(guān)閉連接藻肄,默認(rèn)為0,即不關(guān)閉存活時(shí)長(zhǎng)較長(zhǎng)的連接

            //命令執(zhí)行失敗時(shí)的重試策略
            MaxRetries:      0,                      // 命令執(zhí)行失敗時(shí)拒担,最多重試多少次嘹屯,默認(rèn)為0即不重試
            MinRetryBackoff: 8 * time.Millisecond,   //每次計(jì)算重試間隔時(shí)間的下限,默認(rèn)8毫秒从撼,-1表示取消間隔
            MaxRetryBackoff: 512 * time.Millisecond, //每次計(jì)算重試間隔時(shí)間的上限州弟,默認(rèn)512毫秒,-1表示取消間隔

        })
        pong, err := GlobalRedisClient.Ping(context.Background()).Result()
        if err != nil {
            log.Fatal(fmt.Errorf("redis connect error:%s", err))
        }
        log.Println(pong)
    })
    return GlobalRedisClient
}

分布式鎖實(shí)現(xiàn)

package goredislock

import (
    "context"
    "log"
    "time"

    "github.com/go-redis/redis/v8"
)

const defaultExpireTime = time.Second * 10

// 分布式鎖
type Locker struct {
    key        string        // redis key
    unlock     bool          // 是否已經(jīng)解鎖 低零,解鎖則不用續(xù)租
    incrScript *redis.Script // lua script
    option     options       // 可選項(xiàng)
}

type Options func(o *options)

type options struct {
    expire      time.Duration // expiration time ,默認(rèn)是10秒
    redisClient *redis.Client // redis 實(shí)例婆翔, 可以傳, 不傳就得用全局的
    ctx         context.Context
}

// 續(xù)租的時(shí)候 獲得和設(shè)置過(guò)期原子操作.
const incrLua = `
if redis.call('get', KEYS[1]) == ARGV[1] then
  return redis.call('expire', KEYS[1],ARGV[2])              
 else
   return '0'                   
end`

func NewLocker(key string, opts ...Options) *Locker {
    var lock = &Locker{
        key:        key,
        incrScript: redis.NewScript(incrLua),
    }
    for _, opt := range opts {
        opt(&lock.option)
    }
    // 沒(méi)設(shè)置過(guò)期時(shí)間
    if lock.option.expire == 0 {
        lock.option.expire = defaultExpireTime
    }
    // 未設(shè)置redis 實(shí)例
    if lock.option.redisClient == nil {
        lock.option.redisClient = GlobalRedisClient
    }
    // 未設(shè)置context
    if lock.option.ctx == nil {
        lock.option.ctx = context.Background()
    }
    return lock
}

// 過(guò)期選項(xiàng)
func WithExpire(expire time.Duration) Options {
    return func(o *options) {
        o.expire = expire
    }
}

// redisClient 選項(xiàng),可以每次都傳毁兆,如果沒(méi)傳浙滤,就用全局都
func WithRedisClient(redisClient *redis.Client) Options {
    return func(o *options) {
        o.redisClient = redisClient
    }
}

func WithContext(ctx context.Context) Options {
    return func(o *options) {
        o.ctx = ctx
    }
}

// 第一個(gè)返回:返回鎖,方便鏈?zhǔn)讲僮?// 第二個(gè) 返回結(jié)果
func (this *Locker) Lock() (*Locker, bool) {
    boolcmd := this.option.redisClient.SetNX(context.Background(), this.key, "1", this.option.expire)
    if ok, err := boolcmd.Result(); err != nil || !ok {
        return this, false
    }
    this.expandLockTime()
    return this, true
}

// 續(xù)租
func (this *Locker) expandLockTime() {
    sleepTime := this.option.expire * 2 / 3
    go func() {
        for {
            time.Sleep(sleepTime)
            if this.unlock {
                break
            }
            this.resetExpire()
        }
    }()
}

// 重新設(shè)置過(guò)期時(shí)間
func (this *Locker) resetExpire() {
    cmd := this.incrScript.Run(this.option.ctx, this.option.redisClient, []string{this.key}, 1, this.option.expire.Seconds())
    v, err := cmd.Result()
    log.Printf("key=%s ,續(xù)期結(jié)果:%v,%v\n", this.key, err, v)
}

// 釋放鎖  干完活后釋放鎖
func (this *Locker) Unlock() {
    this.unlock = true
    this.option.redisClient.Del(this.option.ctx, this.key)
}

倉(cāng)庫(kù)地址: https://github.com/cr-mao/goredislock

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末气堕,一起剝皮案震驚了整個(gè)濱河市纺腊,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌茎芭,老刑警劉巖揖膜,帶你破解...
    沈念sama閱讀 218,386評(píng)論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異梅桩,居然都是意外死亡壹粟,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,142評(píng)論 3 394
  • 文/潘曉璐 我一進(jìn)店門(mén)宿百,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)趁仙,“玉大人,你說(shuō)我怎么就攤上這事垦页∪阜眩” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 164,704評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵痊焊,是天一觀的道長(zhǎng)盏袄。 經(jīng)常有香客問(wèn)我忿峻,道長(zhǎng),這世上最難降的妖魔是什么辕羽? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,702評(píng)論 1 294
  • 正文 為了忘掉前任逛尚,我火速辦了婚禮,結(jié)果婚禮上刁愿,老公的妹妹穿的比我還像新娘绰寞。我一直安慰自己,他們只是感情好铣口,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,716評(píng)論 6 392
  • 文/花漫 我一把揭開(kāi)白布克握。 她就那樣靜靜地躺著,像睡著了一般枷踏。 火紅的嫁衣襯著肌膚如雪菩暗。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 51,573評(píng)論 1 305
  • 那天旭蠕,我揣著相機(jī)與錄音停团,去河邊找鬼。 笑死掏熬,一個(gè)胖子當(dāng)著我的面吹牛佑稠,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播旗芬,決...
    沈念sama閱讀 40,314評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼舌胶,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了疮丛?” 一聲冷哼從身側(cè)響起幔嫂,我...
    開(kāi)封第一講書(shū)人閱讀 39,230評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎誊薄,沒(méi)想到半個(gè)月后履恩,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,680評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡呢蔫,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,873評(píng)論 3 336
  • 正文 我和宋清朗相戀三年切心,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片片吊。...
    茶點(diǎn)故事閱讀 39,991評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡绽昏,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出俏脊,到底是詐尸還是另有隱情全谤,我是刑警寧澤,帶...
    沈念sama閱讀 35,706評(píng)論 5 346
  • 正文 年R本政府宣布联予,位于F島的核電站啼县,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏沸久。R本人自食惡果不足惜季眷,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,329評(píng)論 3 330
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望卷胯。 院中可真熱鬧子刮,春花似錦、人聲如沸窑睁。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,910評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)担钮。三九已至橱赠,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間箫津,已是汗流浹背狭姨。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,038評(píng)論 1 270
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留苏遥,地道東北人饼拍。 一個(gè)月前我還...
    沈念sama閱讀 48,158評(píng)論 3 370
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像田炭,于是被迫代替她去往敵國(guó)和親师抄。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,941評(píng)論 2 355

推薦閱讀更多精彩內(nèi)容