亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Go語言中通過Lua腳本操作Redis的方法

 更新時間:2018年01月28日 10:20:25   作者:吃土的漢子  
這篇文章主要給大家介紹了關(guān)于Go語言中通過Lua腳本操作Redis的方法,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考借鑒,下面隨著小編來一起學習學習吧。

前言

為了在我的一個基本庫中降低與Redis的通訊成本,我將一系列操作封裝到LUA腳本中,借助Redis提供的EVAL命令來簡化操作。

EVAL能夠提供的特性:

  • 可以在LUA腳本中封裝若干操作,如果有多條Redis指令,封裝好之后只需向Redis一次性發(fā)送所有參數(shù)即可獲得結(jié)果
  • Redis可以保證Lua腳本運行期間不會有其他命令插入執(zhí)行,提供像數(shù)據(jù)庫事務一樣的原子性
  • Redis會根據(jù)腳本的SHA值緩存腳本,已經(jīng)緩存過的腳本不需要再次傳輸Lua代碼,減少了通信成本,此外在自己代碼中改變Lua腳本,執(zhí)行時Redis必定也會使用最新的代碼。

導入常見的Go庫如 "github.com/go-redis/redis",就可以實現(xiàn)以下代碼。

生成一段Lua腳本

// KEYS: key for record
// ARGV: fieldName, currentUnixTimestamp, recordTTL
// Update expire field of record key to current timestamp, and renew key expiration
var updateRecordExpireScript = redis.NewScript(`
redis.call("EXPIRE", KEYS[1], ARGV[3])
redis.call("HSET", KEYS[1], ARGV[1], ARGV[2])
return 1
`)

該變量創(chuàng)建時,Lua代碼不會被執(zhí)行,也不需要有已存的Redis連接。

Redis提供的Lua腳本支持,默認有KEYS、ARGV兩個數(shù)組,KEYS代表腳本運行時傳入的若干鍵值,ARGV代表傳入的若干參數(shù)。由于Lua代碼需要保持簡潔,難免難以讀懂,最好為這些參數(shù)寫一些注釋

注意:上面一段代碼使用``跨行,`所在的行雖然空白回車,也會被認為是一行,報錯時不要看錯代碼行號。

運行一段Lua腳本

 updateRecordExpireScript.Run(c.Client, []string{recordKey(key)}, 
         expireField,
         time.Now().UTC().UnixNano(), int64(c.opt.RecordTTL/time.Second)).Err()

運行時,Run將會先通過EVALSHA嘗試通過緩存運行腳本。如果沒有緩存,則使用EVAL運行,這時Lua腳本才會被整個傳入Redis。

Lua腳本的限制

  • Redis不提供引入額外的包,例如os等,只有redis這一個包可用。
  • Lua腳本將會在一個函數(shù)中運行,所有變量必須使用local聲明
  • return返回多個值時,Redis將會只給你第一個

腳本中的類型限制

  • 腳本返回nil時,Go中得到的是err = redis.Nil(與Get找不到值相同)
  • 腳本返回false時,Go中得到的是nil,腳本返回true時,Go中得到的是int64類型的1
  • 腳本返回{"ok": ...}時,Go中得到的是redis的status類型(true/false)
  • 腳本返回{"err": ...}時,Go中得到的是err值,也可以通過return redis.error_reply("My Error")達成
  • 腳本返回number類型時,Go中得到的是int64類型
  • 傳入腳本的KEYS/ARGV中的值一律為string類型,要轉(zhuǎn)換為數(shù)字類型應當使用to_number

如果腳本運行了很久會發(fā)生什么?

Lua腳本運行期間,為了避免被其他操作污染數(shù)據(jù),這期間將不能執(zhí)行其它命令,一直等到執(zhí)行完畢才可以繼續(xù)執(zhí)行其它請求。當Lua腳本執(zhí)行時間超過了lua-time-limit時,其他請求將會收到Busy錯誤,除非這些請求是SCRIPT KILL(殺掉腳本)或者SHUTDOWN NOSAVE(不保存結(jié)果直接關(guān)閉Redis)

更多內(nèi)容參考以下地址,我這里主要是根據(jù)使用Go的經(jīng)驗提供一些總結(jié)。https://redis.io/commands/eval

一段更“復雜”的腳本,它要求在獲取一個key值時,如果該值訪問較多,就延長生存周期。此外還要比較更新時間,如果不需要更新,則直接返回取到的值,否則返回redis.Nil

// KEYS: rec:key, key
// ARGV: currentUnixTimestamp, hotHit, recordTTL, ttl
// When there's a hit,
var fetchRecordScript = redis.NewScript(local value = redis.call("GET", KEYS[2]) if(value == nil) then return nil end local hit = redis.call("HINCRBY", KEYS[1], "hit", 1) redis.call("EXPIRE", KEYS[1], ARGV[3]) local minHotHit = tonumber(ARGV[2]) local keyTTL = tonumber(ARGV[4]) if(hit > minHotHit)then keyTTL = keyTTL * 2 end redis.call("EXPIRE", KEYS[2], keyTTL) local expire = tonumber(redis.call("HGET", KEYS[1], "expire")) local unixTime = tonumber(ARGV[1]) if(expire == nil or expire < unixTime) then return nil else return value end)
// KEYS: key for record
// ARGV: fieldName, currentUnixTimestamp, recordTTL
// Update expire field of record key to current timestamp, and renew key expiration
var updateRecordExpireScript = redis.NewScript(redis.call("EXPIRE", KEYS[1], ARGV[3]) redis.call("HSET", KEYS[1], ARGV[1], ARGV[2]) return 1)

總結(jié)

以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對腳本之家的支持。

相關(guān)文章

  • Go語言中如何實現(xiàn)并發(fā)

    Go語言中如何實現(xiàn)并發(fā)

    Go的并發(fā)機制通過協(xié)程和通道的簡單性和高效性,使得編寫并發(fā)代碼變得相對容易,這種并發(fā)模型被廣泛用于構(gòu)建高性能的網(wǎng)絡服務、并行處理任務和其他需要有效利用多核處理器的應用程序,這篇文章主要介紹了在Go中如何實現(xiàn)并發(fā),需要的朋友可以參考下
    2023-09-09
  • GoLang并發(fā)機制探究goroutine原理詳細講解

    GoLang并發(fā)機制探究goroutine原理詳細講解

    goroutine是Go語言提供的語言級別的輕量級線程,在我們需要使用并發(fā)時,我們只需要通過 go 關(guān)鍵字來開啟 goroutine 即可。這篇文章主要介紹了GoLang并發(fā)機制goroutine原理,感興趣的可以了解一下
    2022-12-12
  • 詳解Go語言中的Slice鏈式操作

    詳解Go語言中的Slice鏈式操作

    這篇文章主要為大家詳細介紹了Go語言中的Slice鏈式操作的相關(guān)知識,文中的示例代碼講解詳細,具有一定的學習價值,感興趣的小伙伴可以了解一下
    2023-04-04
  • Golang并發(fā)操作中常見的讀寫鎖詳析

    Golang并發(fā)操作中常見的讀寫鎖詳析

    Golang中的鎖機制主要包含互斥鎖和讀寫鎖互斥鎖互斥鎖是傳統(tǒng)并發(fā)程序?qū)蚕碣Y源進行控制訪問的主要手段,這篇文章主要給大家介紹了關(guān)于Golang并發(fā)操作中常見的讀寫鎖的相關(guān)資料,需要的朋友可以參考下
    2021-08-08
  • 初識Golang?Mutex互斥鎖的使用

    初識Golang?Mutex互斥鎖的使用

    在學習操作系統(tǒng)的時候,我們應該都學習過臨界區(qū)、互斥鎖這些概念,用于在并發(fā)環(huán)境下保證狀態(tài)的正確性。在?Go語言?里面互斥鎖是?sync.Mutex?,我們本篇文章就來學習下為什么要使用互斥鎖、如何使用互斥鎖,以及使用時的常見問題
    2022-10-10
  • Go 語言中 20 個占位符的整理

    Go 語言中 20 個占位符的整理

    這篇文章主要介紹了Go 語言中 20 個占位符的整理,看完本篇文章講學會什么是占位符?哪些函數(shù)支持?如何使用占位符?不同的占位符的作用?配合占位符的幾個標記符號用法?
    2021-10-10
  • go語言中讀取配置文件的方法總結(jié)

    go語言中讀取配置文件的方法總結(jié)

    這篇文章主要為大家詳細介紹了go語言中讀取配置文件的幾個常見方法,文中的示例代碼講解詳細,具有一定的借鑒價值,需要的小伙伴可以參考下
    2023-08-08
  • golang中日期操作之日期格式化及日期轉(zhuǎn)換

    golang中日期操作之日期格式化及日期轉(zhuǎn)換

    在編程中,程序員會經(jīng)常使用到日期相關(guān)操作,下面這篇文章主要給大家介紹了關(guān)于golang中日期操作之日期格式化及日期轉(zhuǎn)換的相關(guān)資料,文中通過實例代碼介紹的非常詳細,需要的朋友可以參考下
    2022-11-11
  • Golang并發(fā)編程重點講解

    Golang并發(fā)編程重點講解

    這篇文章主要介紹了Golang并發(fā)編程,在許多環(huán)境中,實現(xiàn)對共享變量的正確訪問所需要的微妙之處使并發(fā)編程變得困難。Go鼓勵一種不同的方法,在這種方法中,共享值在通道中傳遞,實際上,從不由單獨的執(zhí)行線程主動共享
    2023-04-04
  • 深入學習Golang的流程控制

    深入學習Golang的流程控制

    Go 語言是一門現(xiàn)代化的編程語言,以其簡潔高效、并發(fā)安全等特點被越來越多的開發(fā)者所使用。本文將深入探討 Go 語言中的流程控制,包括條件語句、循環(huán)語句以及控制語句等方面
    2023-05-05

最新評論