一般來說,應(yīng)用程序處理業(yè)務(wù)的邏輯,是將執(zhí)行代碼發(fā)送到數(shù)據(jù)庫。因此每次執(zhí)行函數(shù)時,代碼都會從客戶端流入服務(wù)器,結(jié)果就是這個過程十分緩慢,甚至?xí)霈F(xiàn)代碼在應(yīng)用程序之間重復(fù)執(zhí)行的情況。
而我們一向是反對復(fù)雜性的!從Redis Gears到Redis7.0再到Redis7.2,我們的愿景就是為開發(fā)人員提供更靈活、高效的解決方案。
Redis Gears:四年前推出的RedisGears,是Redis平臺內(nèi)的第一個可編程模型,開發(fā)人員能在數(shù)據(jù)所在的地方編寫并執(zhí)行腳本。
Redis 7.0:Redis 7.0引入了帶有函數(shù)的腳本方法,函數(shù)作為數(shù)據(jù)庫的一部分,提高了可用性和持久性,繼承了數(shù)據(jù)的復(fù)制和持久性級別。
Redis 7.2:Redis 7.2更進一步,引入了觸發(fā)器和函數(shù),以增強Redis的可編程性;擴展了服務(wù)器端功能;改進了數(shù)據(jù)庫中執(zhí)行功能的方式和時間;并促進了直接在數(shù)據(jù)所在的地方執(zhí)行復(fù)雜的業(yè)務(wù)邏輯。
01
觸發(fā)器和函數(shù)的基礎(chǔ)知識
觸發(fā)器和函數(shù)是通過Redis Stack提供的新一代可編程功能。它允許開發(fā)人員直接在Redis數(shù)據(jù)庫中針對數(shù)據(jù)更改進行編程、存儲和自動執(zhí)行JavaScript代碼。
1、開發(fā)人員可以定義事件(稱為觸發(fā)器)來執(zhí)行更接近數(shù)據(jù)的函數(shù)。也就是說,開發(fā)人員定義響應(yīng)數(shù)據(jù)庫事件或命令而執(zhí)行的業(yè)務(wù)邏輯。這加快了代碼和相關(guān)交互的速度,因為無需等待即可將代碼從客戶端引入數(shù)據(jù)庫。
2、加快對Redis中其他事件的反應(yīng)時間,如鍵空間通知,這些通知無法通過發(fā)布和訂閱(Pub/Sub)事件等其他方式實時處理。
3、遠(yuǎn)程功能:觸發(fā)器和函數(shù)處理集群數(shù)據(jù)庫內(nèi)的分布,在每個分片上安裝庫并根據(jù)密鑰所在的位置執(zhí)行函數(shù)。遠(yuǎn)程函數(shù)允許執(zhí)行讀取操作,這些操作可以從任何槽訪問數(shù)據(jù),甚至在集群數(shù)據(jù)庫中也是如此,因此可以從每個函數(shù)訪問所有數(shù)據(jù)。
02
編寫語言:從Lua到JavaScript
Redis使用Lua來編寫腳本和函數(shù)。Lua有很多好處,例如代碼可重用性,但它并不是專業(yè)開發(fā)人員常用的語言。根據(jù)2022 年StackOverflow開發(fā)者調(diào)查,只有3.2%的開發(fā)人員在專業(yè)領(lǐng)域使用 Lua。
因此,在Redis7.2中引入了JavaScript,這種更流行的語言的引入,可以降低新Redis開發(fā)人員的采用成本。
03
應(yīng)用程序代碼更容易維護
觸發(fā)器和函數(shù)的另一個好處是它降低了跨多個應(yīng)用程序管理業(yè)務(wù)邏輯的復(fù)雜性。
當(dāng)多個應(yīng)用程序訪問同一數(shù)據(jù)庫時,開發(fā)人員必須協(xié)調(diào)應(yīng)用程序如何以一致的方式處理數(shù)據(jù)。在每個應(yīng)用程序中復(fù)制代碼以驗證數(shù)據(jù)、豐富搜索結(jié)果,或在另一個應(yīng)用程序進行更改時更新數(shù)據(jù)庫也是很常見的。
有了觸發(fā)器和函數(shù),就不再需要在多個應(yīng)用程序上重復(fù)代碼。代碼始終以相同的方式執(zhí)行,按需執(zhí)行或由數(shù)據(jù)庫中的事件啟動。
04
數(shù)據(jù)庫事件實時處理
到目前為止,開發(fā)人員對Redis中的數(shù)據(jù)庫事件做出反應(yīng)需要依賴Pub/Sub機制。雖然 Pub/Sub有很多優(yōu)點,但它并不總是正確的選擇。特別是,Pub/Sub不是實時的??蛻舳吮仨氈鲃颖O(jiān)聽事件;如果客戶端沒有監(jiān)聽,事件就會丟失。
在Redis7.2時代,開發(fā)人員可以注冊基于鍵前綴和事件類型執(zhí)行的鍵空間觸發(fā)器。觸發(fā)器可以以原子方式執(zhí)行,因此在事件和業(yè)務(wù)邏輯之間不處理其他Redis事件。
05
如何操作觸發(fā)器和函數(shù)
這里,我們介紹如何注冊一個函數(shù)和一個觸發(fā)器。當(dāng)通過TFCALL命令調(diào)用函數(shù)時,函數(shù)就會被執(zhí)行;觸發(fā)器根據(jù)Redis中的事件執(zhí)行。
1.序言定義我們使用js引擎,庫名稱為lib,所需的觸發(fā)器和函數(shù)API的最低版本為1.0。
#!js name=lib api_version=1.0
2.接下來,我們創(chuàng)建一個返回Redis命令結(jié)果的函數(shù)??蛻舳嗽试S在我們的函數(shù)中執(zhí)行Redis 命令。數(shù)據(jù)事件包含運行函數(shù)時可以提供的鍵和參數(shù)。
function answer(client, data) { return client.call(“ping”);}
3.Redis全局變量允許我們注冊觸發(fā)器和函數(shù),并記錄到日志文件中。我們?yōu)樵摵瘮?shù)注冊一個名稱,并在函數(shù)執(zhí)行時調(diào)用它。
redis.registerFunction(‘playPingPong’, answer);
4.完整的JavaScript文件如下所示,將其另存為lib.js.
#!js name=lib api_version=1.0 function answer(client, data) { return client.call(‘ping’);} redis.registerFunction(‘playPingPong’, answer);
5.然后我們使用命令在觸發(fā)器和函數(shù)中注冊我們的函數(shù)TFUNCTION LOAD。該TFUNCTION LOAD命令還可以在集群數(shù)據(jù)庫中分發(fā)函數(shù)庫。
> redis-cli -x TFUNCTION LOAD < ./lib.jsOK
6.現(xiàn)在我們可以使用TFCALL命令執(zhí)行該函數(shù)。該命令獲取以句點分隔的庫名稱和函數(shù)名稱。
>redis-cli TFCALL lib.playPingPong 0“PONG”
7.這樣,就成功地在Redis數(shù)據(jù)庫中創(chuàng)建、注冊并觸發(fā)了一個函數(shù)。
8.我們可以使用鍵空間觸發(fā)器拓展這個示例。我們添加了一個新的注冊,該注冊對前綴為'fellowship:'的鍵做出反應(yīng)。在lib.js文件末尾添加此代碼。
function addLastUpdatedField(client, data) { if(data.event == ‘hset’) { var currentDateTime = Date.now(); client.call(‘hset’, data.key, ‘last_updated’, currentDateTime.toString());}} redis.registerKeySpaceTrigger(‘a(chǎn)ddLastUpdated’, 'fellowship:', addLastUpdatedField);
9. 使用帶有 REPLACE 參數(shù)的 TFUNCTION LOAD 命令來更新現(xiàn)有庫。該TFUNCTION LOAD REPLACE命令立即更新所有使用 Redis 數(shù)據(jù)庫的客戶端,并且它們會開始使用新的業(yè)務(wù)邏輯。
>redis-cli -x TFUNCTION LOAD REPLACE . < ./lib.jsOK
10.要測試新的鍵空間觸發(fā)器,請創(chuàng)建一個以Fellowship:開頭的新鍵,并使用RedisInsight 檢查字段。鍵空間觸發(fā)器是通過命令執(zhí)行的,因此在創(chuàng)建鍵時已經(jīng)添加了last_updated字段。最后,我們可以在RedisInsight中查看結(jié)果。
-
數(shù)據(jù)
+關(guān)注
關(guān)注
8文章
7122瀏覽量
89349 -
觸發(fā)器
+關(guān)注
關(guān)注
14文章
2002瀏覽量
61279 -
函數(shù)
+關(guān)注
關(guān)注
3文章
4344瀏覽量
62849
發(fā)布評論請先 登錄
相關(guān)推薦
評論