亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 數據庫 > Redis > 正文

簡介Lua腳本與Redis數據庫的結合使用

2020-03-17 12:41:59
字體:
來源:轉載
供稿:網友
這篇文章主要介紹了簡介Lua腳本與Redis數據庫的結合使用,Redis是基于主存的高性能數據庫,需要的朋友可以參考下
 

 可能你已經聽說過Redis 中嵌入了腳本語言,但是你還沒有親自去嘗試吧?  這個入門教程會讓你學會在你的Redis 服務器上使用強大的lua語言。
Hello, Lua!

我們的第一個Redis Lua 腳本僅僅返回一個字符串,而不會去與redis 以任何有意義的方式交互。

 

復制代碼代碼如下:
local msg = "Hello, world!"
return msg

 

這是非常簡單的,第一行代碼定義了一個本地變量msg存儲我們的信息, 第二行代碼表示 從redis 服務端返回msg的值給客戶端。 保存這個文件到hello.lua,像這樣去運行:

 

復制代碼代碼如下:
redis-cli EVAL "$(cat hello.lua)" 0

 

運行這段代碼會打印"Hello,world!", EVAL在第一個參數是我們的lua腳本, 這我們用cat命令從文件中讀取我們的腳本內容。第二個參數是這個腳本需要訪問的Redis 的鍵的數字號。我們簡單的 “Hello Script" 不會訪問任何鍵,所以我們使用0

訪問鍵和參數

假設我們要建立一個URL簡寫服務器。我們就要去存儲每條進入的URL并返回一個唯一數值,以便以后通過這個數值訪問到該URL。

我們將利用Lua腳本立即從Redis中用INCRand獲取一個唯一標識ID,以這個標識ID作為URL存儲于一個哈希中的鍵值:

 

復制代碼代碼如下:
local link_id = redis.call("INCR", KEY[1])
redis.call("HSET", KEYS[2], link_id, ARGV[1])
return link_id

 

我們將用call()函數首次訪問Redis。call()的參數就是發給Redis的命令:首先INCR <key>, 然后HSET <key> <field> <value>。這兩個命令將依次執行——當這個腳本執行時,Redis不會做任何事,它將非??斓剡\行。

我們將會訪問兩個Lua表:KEYS和ARGV。表單是關聯性數組和結構化數據的Lua唯一機制。對于我們的意圖,你可以把它們看做是一個你所熟悉的任意語言對等的數組,但是提醒兩個很容易困擾到新手的兩個Lua定則:

  •     表是基于1的,也就是說索引以數值1開始。所以在表中的第一個元素就是mytable[1],第二個就是mytable[2]等等。
  •     表中不能有nil值。如果一個操作表中有[1, nil, 3, 4],那么結果將會是[1]——表將會在第一個nil截斷。

當調用這個腳本時,我們還需要傳遞KEYS和ARGV表的值:

 

復制代碼代碼如下:
redis-cli EVAL "$(cat incr-and-stor.lua)" 2 links:counter links:urls http://malcolmgladwellbookgenerator.com/

 


在EVAL語句中,2指出需要傳入的KEY的個數,后面跟著需要傳入的兩個KEY,最后傳入是ARGV的值。在Redis中執行Lua腳本時,Redis-cli會檢查傳入KEY的個數,除非傳入的完全是命令。

為了解釋得更清楚,下面列出替換KEY和ARGV后的腳本:

 

復制代碼代碼如下:
local link_id = redis.call("INCR", "links:counter")
redis.call("HSET", "links:urls", link_id, "http://malcolmgladwellbookgenerator.com")
return link_id

 

為Redis編寫Lua腳本時,每個KEY都是通過KEYS表指定。ARGV表用來傳遞參數,這個例子中ARGV用來傳入URL。

邏輯條件:increx與hincrex

上一個例子保存鏈接為短網址,想要知道這個鏈接的點擊次數,在Redis中添加一個hash計數器。當帶有鏈接標記的用戶訪問時,我們檢查其是否存在,如存在則需要給計數器加1:

 

復制代碼代碼如下:
if redis.call("HEXISTS", KEYS[1], ARGV[1]) == 1 then
return redis.call("HINCR", KEYS[1], ARGV[1])
else
return nil
end

 

每次有人點擊短網址,我們運行這個腳本跟蹤這個鏈接被再次分享。我們用EVAL來調用腳本,傳入inlinks:visits(keys[1])和上一個腳本返回的鏈接標識(ARGV[1])。

這段腳本將檢查是否存在相同的hash,如果存在就為這個標準的Redis KEY加1。

 

復制代碼代碼如下:
if redis.call("EXISTS",KEYS[1]) == 1 then
return redis.call("INCR",KEYS[1])
else
return nil
end

 

腳本加載與注冊執行

注意,當Redis在運行Lua腳本的時候,其它的事情什么都干不了!腳本最好只是簡單的擴展Redis進行較小的原子操作和簡單的邏輯控制需要,Lua腳本中的bug可能引發整個Redis服務器鎖—最好保持腳本的簡短和易于調試。

雖然這些腳本一般都比較短小,但我們還是希望不要每次執行時都使用完整的Lua腳本,實際上可以在程序一步一步(譯注:application boots翻譯有難度)開發中注冊Lua腳本(或者在你部署時注冊),然后用注冊后生成的SHA-1標識來進行調用。

 

復制代碼代碼如下:
redis-cli SCRIPT LOAD "return 'hello world'"
=> "5332031c6b470dc5a0dd9b4bf2030dea6d65de91"
 
redis-cli EVALSHA 5332031c6b470dc5a0dd9b4bf2030dea6d65de91 0
=> "hello world"

 


顯示調用SCRIPT LOAD通常是不必要的,當一個程序執行EVAL時就已隱式加載了。程序會先嘗試EAVALSHA,當腳本沒有找到時會調用EVAL。

對于Ruby開發者,可以看一下Shopify's Wolverine,其可以為Ruby應用簡單的加載并存儲Lua腳本。對于PHP開發者,Predis 支持加載Lua腳本作為普通Redis命令進行調用(譯注:需要繼承Predis/Command/ScriptedCommand基類,并注冊命令)。如果你使用這些或者其它的工具來標準化與Lua的交互,請讓我知道,我很感興趣知道本文之外的內容。

何時使用Lua

Redis支持WATCH/MULTI/EXEC這樣的塊,能進行一組操作,也能一起提交執行,看起來與Lua有重疊。應該如何進行選擇?MULT塊中所有操作獨立,但在Lua中,后面的操作能依賴前面操作的執行結果。同時使用Lua腳本還能夠避免WATCH使用后競爭條件引起客戶端反應變慢的情況。

在RedisGreen(譯注:國外一家專門提供Redis主機的服務商),我們看到許多應用使用Lua的同時也使用MULTI/EXEC,但兩者但不是替代關系。許多成功的Lua腳本都很小,僅僅實現一個你的應用需要而Redis命令中沒有單一的功能。

訪問庫

Redis的Lua解釋器加載七個庫:base,table,string, math, debug,cjson和cmsgpack。前幾個都是標準庫,充許你使用任何語言進行基本的操作。后面兩個可以讓Redis支持JSON和MessagePack—這是非常有用的功能,同時我也很想知道為什么常??床坏竭@種用法。

Web應用程序常常使用JSON作為api返回數據,你也許也可以把一堆JSON數據存到Redis的key中。當想訪問某些JSON數據時,首先需要保存到一個hash中,使用Redis的JSON支持將非常方便:

 

復制代碼代碼如下:
if redis.call("EXISTS", KEYS[1]) == 1 then
local payload = redis.call("GET", KEYS[1])
return cjson.decode(payload)[ARGV[1]]
else
return nil
end

 


在這里我們檢查看key是否存在,如不存在則快速返回nil。如存在則從Redis中獲取JSON值,用cjson.decode()進行解析,然后返回請求內容。

 

復制代碼代碼如下:
redis-cli set apple '{ "color": "red", "type": "fruit" }'
=> OK
 
redis-cli eval "$(cat json-get.lua)" 1 apple type
=> "fruit"

 

加載這段腳本進你的Redis服務器,將JSON數據保存到Redis中,通常是hash。 雖然我們每次訪問時都必須解析,但只要你的對象很小,這個操作實際上是非常快的。

如果你的API只是在內部提供,通常需要考慮效率上的問題,MessagePack 是比采用JSON更好的選擇,它更小,更快,在Redis(更多場合也是如此),MessagePack是JSON更好的替代品。

 

復制代碼代碼如下:
if redis.call("EXISTS", KEYS[1]) == 1 then
  local payload = redis.call("GET", KEYS[1])
  return cmsgpack.unpack(payload)[ARGV[1]]
else
  return nil
end

 


數值轉換

Lua和Redis各有自己的一套類型,因此,理解Redis與Lua在邊界調用相互轉換引起值的改變是非常重要的。一個來自Lua中number返回到Redis客戶端時變成了integer—任何數字后面的小數點都被清除了:

 

復制代碼代碼如下:
local indiana_pi = 3.2
return indiana_pi

 

在你運行這段腳本時,Redis將返回一個整數3,丟失了pi中有用的片段。看起來很簡單,但是一旦開始進行Redis與中間腳本交互時就需要更小心。例如:

 

復制代碼代碼如下:
local indiana_pi = 3.2
redis.call("SET", "pi", indiana_pi)
return redis.call("GET", "pi")

 

執行的結果是一個字符串:“3.2”,這是為什么呢?在Redis中沒有專有的數值類型,當我們第一次調用SET的時候,Redis就已經將它保存為字符串了,將Lua初始化時將其作為一個浮點數的類型信息給丟失了。所以當我們后面取出這個值時,它就變成了一個字符串。

在Redis中,除了INCR和DECR,其它的GET,SET操作所訪問的數據都作為字符串處理。INCR與DECR是專門對數值的操作,實際上返回是整數(integer)回復(維護和存儲遵守數字規則),但Redis內部保存類型實際上還是字符串值。

總結:

下面這些都是在Redis中使用Lua時常見的錯誤:

  •     表是Lua中的表達式,與很多流行語言不同。KEYS中的第一個元素是KEYS[1],第二個是KEYS[2](譯注:不是0開始)
  •     nil是表的結束符,[1,2,nil,3]將自動變為[1,2],因此在表中不要使用nil。
  •     redis.call會觸發Lua中的異常,redis.pcall將自動捕獲所有能檢測到的錯誤并以表的形式返回錯誤內容。
  •     Lua數字都將被轉換為整數,發給Redis的小數點會丟失,返回前把它們轉換成字符串類型。
  •     確保在Lua中使用的所有KEY都在KEY表中,否則在將來的Redis版中你的腳本都有不能被很好支持的危險。
  •     Lua腳本和其它Redis操作一樣,在腳本執行時,其它的一切都不能運行??紤]用腳本來護展Redis服務器能力,但要保持短小和有用。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
色噜噜久久综合伊人一本| 日韩欧亚中文在线| 欧美激情一级精品国产| 亚洲伊人成综合成人网| 国产精品久久久久久久久久ktv| 97精品国产97久久久久久春色| 亚洲人成自拍网站| 国内精品久久久久久久久| 精品福利视频导航| 日韩高清电影好看的电视剧电影| 精品国产欧美一区二区三区成人| 国产精品 欧美在线| 色一区av在线| 国产亚洲视频在线观看| 日本伊人精品一区二区三区介绍| 日韩网站在线观看| 亚洲女人被黑人巨大进入| 亚洲美女又黄又爽在线观看| 7777免费精品视频| 日韩第一页在线| 国产成人精品久久二区二区| 欧美成人手机在线| 57pao成人国产永久免费| 亚洲成人在线视频播放| 国产一区av在线| 久久久久久久久爱| 亚洲国产91精品在线观看| 日韩美女视频免费在线观看| 国产欧美精品一区二区三区介绍| 亚洲精品天天看| 国产精品久久婷婷六月丁香| 国产精品网站入口| 久久久精品久久久| 精品成人乱色一区二区| 亚洲开心激情网| 91久久中文字幕| 国产精品高潮呻吟久久av无限| 欧美裸体xxxxx| 久久久国产视频| 精品久久久久久国产91| 丝袜一区二区三区| 久久国产精品久久久| 亚洲大尺度美女在线| 久久久久国产精品www| 精品无人国产偷自产在线| 成人a级免费视频| 亚洲成年网站在线观看| 久久99亚洲热视| 久热精品视频在线观看一区| 欧美性做爰毛片| 亚洲国产精品福利| 丝袜亚洲另类欧美重口| 精品久久久久久久久久久久| 亚洲国产欧美自拍| 久久亚洲精品国产亚洲老地址| 日韩av网址在线观看| 久久资源免费视频| 欧美性生交大片免费| 国产精品视频xxx| 欧美精品在线播放| 日韩av大片在线| 91久热免费在线视频| 国产91热爆ts人妖在线| 日韩中文有码在线视频| 91久久久久久国产精品| 欧美在线观看网址综合| 亚洲成人网久久久| 日韩欧美一区二区三区| 欧美日韩一区二区精品| 国产精品稀缺呦系列在线| 欧美日本黄视频| 国产一区二区视频在线观看| 亚洲福利在线视频| 国产精品久久久久久久久久久久久久| 国产视频自拍一区| 日本精品性网站在线观看| 欧美精品生活片| 亚洲欧美国产精品专区久久| 91中文精品字幕在线视频| 国产一区二区免费| 亚洲国产小视频| 亚洲区bt下载| 川上优av一区二区线观看| 国产欧美日韩中文字幕在线| 欧美性猛交xxxxx免费看| 欧美日韩国产中字| 日韩高清有码在线| 国产成人福利视频| 亚洲黄色av女优在线观看| 久久久久久国产| 91久久久久久国产精品| 国产一区二区三区高清在线观看| 日韩美女视频免费在线观看| 人九九综合九九宗合| 国产深夜精品福利| 日韩国产激情在线| 国产精品久久久亚洲| 国产亚洲欧美日韩美女| 亚洲色图偷窥自拍| 久久久久国产精品一区| 亚洲人成亚洲人成在线观看| 欧美在线激情网| 国产精品中文字幕在线观看| 一区二区三区视频免费在线观看| 日本午夜精品理论片a级appf发布| 亚洲一区二区三区久久| 久久九九全国免费精品观看| 欧美日韩国产在线看| 国产一区二区av| 国产成人久久精品| 久久亚洲精品一区二区| 热久久美女精品天天吊色| 日产精品久久久一区二区福利| 亚洲女人初尝黑人巨大| 日韩免费av在线| 国产精品视频在线观看| 欧美日韩国产影院| 日韩av大片免费看| 欧美黑人一级爽快片淫片高清| 91夜夜未满十八勿入爽爽影院| 国产精品久久久久久久av电影| 97涩涩爰在线观看亚洲| 97国产精品视频人人做人人爱| 国产亚洲视频中文字幕视频| 欧美成人一区二区三区电影| 日韩经典一区二区三区| 色久欧美在线视频观看| 成人激情视频小说免费下载| 欧美日韩激情美女| 亚洲成人xxx| 国产精品久久久91| 午夜美女久久久久爽久久| 久久精品91久久香蕉加勒比| 久久久久一本一区二区青青蜜月| 久久99视频精品| 日韩欧美国产黄色| 在线观看日韩视频| 色综合伊人色综合网站| 日韩禁在线播放| 美女av一区二区| 国产一区红桃视频| 欧美黄色片在线观看| 色在人av网站天堂精品| 丁香五六月婷婷久久激情| 日本欧美国产在线| 国产有码一区二区| 欧美一级高清免费播放| 亚洲精品久久久久久久久久久| 欧美一级大片在线免费观看| 久热爱精品视频线路一| 日韩在线中文字幕| 在线观看国产成人av片| 欧美理论电影网| 日韩av电影手机在线| 91日本在线观看| 国产一区二区三区直播精品电影| 欧美日韩一区二区免费在线观看| 97在线免费观看| 亚洲日本欧美日韩高观看| 国产一区二区黑人欧美xxxx| 国产成人精品一区| 国产欧美最新羞羞视频在线观看| 国产精品扒开腿爽爽爽视频| 国产激情999|