第七色在线视频,2021少妇久久久久久久久久,亚洲欧洲精品成人久久av18,亚洲国产精品特色大片观看完整版,孙宇晨将参加特朗普的晚宴

為了賬號(hào)安全,請(qǐng)及時(shí)綁定郵箱和手機(jī)立即綁定
已解決430363個(gè)問題,去搜搜看,總會(huì)有你想問的

使用SETNX的單實(shí)例Redis鎖

使用SETNX的單實(shí)例Redis鎖

Go
慕尼黑5688855 2023-06-26 15:35:22
我需要從應(yīng)用程序客戶端連接到單個(gè) Redis 實(shí)例。由于客戶端將在 Kubernetes 中進(jìn)行復(fù)制,因此我正在研究有關(guān)鎖的 Redis 文檔,以防止客戶端副本之間的競(jìng)爭(zhēng)。有趣的是,SETNX文檔明確建議不要使用SETNX鎖來實(shí)現(xiàn),并指出它基本上已經(jīng)過時(shí)了:為了支持 Redlock 算法,不鼓勵(lì)使用以下模式 [...]無論如何,我們都會(huì)記錄舊模式,因?yàn)槟承┈F(xiàn)有實(shí)現(xiàn)鏈接到此頁面作為參考。然而,Redlock 算法是專門為分布式鎖量身定制的,因此當(dāng)一個(gè)人試圖鎖定多個(gè) Redis 實(shí)例時(shí),它們實(shí)際上是指多個(gè)主實(shí)例。更進(jìn)一步,庫redsync?(golang) 聲明該New函數(shù)如下:func New(pools []Pool) *Redsync {? ? return &Redsync{? ? ? ? pools: pools,? ? }}它看起來毫無疑問是為了支持 Redis 集群上的鎖定而設(shè)計(jì)的。在我的用例中,我將僅連接到一個(gè) Redis 實(shí)例。也許我可以只使用 redsync 包并傳遞長(zhǎng)度為 1 的切片,但對(duì)我來說,該SETNX模式似乎可以在單個(gè) Redis 實(shí)例上同樣正常工作。我的看法正確嗎?謝謝
查看完整描述

2 回答

?
慕無忌1623718

TA貢獻(xiàn)1744條經(jīng)驗(yàn) 獲得超4個(gè)贊

是的,Redlock 算法確實(shí)是為分布式 Redis 系統(tǒng)設(shè)計(jì)的,如果您使用單個(gè)實(shí)例,那么使用 SET 和 SETNX 文檔中描述的更簡(jiǎn)單的鎖定方法應(yīng)該沒問題。

然而,更重要的一點(diǎn)是:您可能不需要使用鎖來避免多個(gè) Redis 客戶端之間的沖突。Redis 鎖通常用于保護(hù)某些外部分布式資源(有關(guān)此問題的更多信息,請(qǐng)參閱我的回答。Redis 本身通常不需要鎖;由于 Redis 的單線程特性,許多命令已經(jīng)是原子的,您可以使用事務(wù)或 Lua 腳本來組成任意復(fù)雜的原子操作。

因此,我的建議是設(shè)計(jì)您的客戶端代碼以使用原子性來避免沖突,而不是嘗試使用鎖(分布式鎖或其他鎖)。


查看完整回答
反對(duì) 回復(fù) 2023-06-26
?
慕仙森

TA貢獻(xiàn)1827條經(jīng)驗(yàn) 獲得超8個(gè)贊

實(shí)現(xiàn) (Go) 如下所示:


// import "github.com/gomodule/redigo/redis"


type redisService struct {

? ? pool? ? ? *redis.Pool

? ? lastLogin *redis.Script // Lua script initialized into this field

}


// Constructing a redis client

func NewRedisService(config *config.RedisClientConfig) RedisService {

? ? return &redisService{

? ? ? ? pool: &redis.Pool{

? ? ? ? ? ? MaxIdle:? ? ?10,

? ? ? ? ? ? IdleTimeout: 120 * time.Second,

? ? ? ? ? ? Dial: func() (redis.Conn, error) {

? ? ? ? ? ? ? ? return redis.Dial("tcp", config.BaseURL)

? ? ? ? ? ? },

? ? ? ? ? ? TestOnBorrow: func(c redis.Conn, t time.Time) error {

? ? ? ? ? ? ? ? if time.Since(t) < time.Minute {

? ? ? ? ? ? ? ? ? ? return nil

? ? ? ? ? ? ? ? }

? ? ? ? ? ? ? ? _, err := c.Do("PING")

? ? ? ? ? ? ? ? return err

? ? ? ? ? ? },

? ? ? ? },

? ? ? ? // initialize Lua script object

? ? ? ? // lastLoginLuaScript is a Go const with the script literal

? ? ? ? lastLogin: redis.NewScript(1, lastLoginLuaScript),

? ? }

}

Lua腳本(注釋解釋了它的作用):


--[[

? ? Check if key exists, if it exists, update the value without changing the remaining TTL.

? ? If it doesn't exist, create it.


? ? Script params

? ? KEYS[1] = the account id used as key

? ? ARGV[1] = the key TTL in seconds

? ? ARGV[2] = the value

]]--

local errorKeyExpired = 'KEXP'

local statusKeyUpdated = 'KUPD'

local statusKeyCreated = 'KCRE'


if redis.call('exists', KEYS[1]) == 1 then

? ? local ttl = redis.call('ttl', KEYS[1])

? ? if ttl < 0 then --[[ no expiry ]]--

? ? ? ? redis.call('setex', KEYS[1], ARGV[1], ARGV[2])

? ? ? ? return redis.status_reply(statusKeyCreated)

? ? end

? ? if ttl == 0 then --[[ expired ]]--

? ? ? ? return redis.error_reply(errorKeyExpired)

? ? end


? ? redis.call('setex', KEYS[1], ttl, ARGV[2])

? ? return redis.status_reply(statusKeyUpdated)


else

? ? redis.call('setex', KEYS[1], ARGV[1], ARGV[2])

? ? return redis.status_reply(statusKeyCreated)

end

用法:


func (rs *redisService) UpsertLastLoginTime(key string, ttl uint, value int64) (bool, error) {

? ? conn := rs.pool.Get()

? ? defer conn.Close()


? ? // call Do on the script object

? ? resp, err := rs.lastLogin.Do(conn, key, ttl, value)


? ? switch resp {

? ? case statusKeyCreated:

? ? ? ? return true, nil


? ? case statusKeyUpdated:

? ? ? ? return false, nil


? ? case errorKeyExpired:

? ? ? ? return false, ErrKeyExpired


? ? default:

? ? ? ? return false, errors.Wrap(err, "script execution error")

? ? }

}


查看完整回答
反對(duì) 回復(fù) 2023-06-26
  • 2 回答
  • 0 關(guān)注
  • 241 瀏覽
慕課專欄
更多

添加回答

舉報(bào)

0/150
提交
取消
微信客服

購(gòu)課補(bǔ)貼
聯(lián)系客服咨詢優(yōu)惠詳情

幫助反饋 APP下載

慕課網(wǎng)APP
您的移動(dòng)學(xué)習(xí)伙伴

公眾號(hào)

掃描二維碼
關(guān)注慕課網(wǎng)微信公眾號(hào)