Redis協程網絡庫 lua-resty-redis
## 簡介
lua-resty-redis 是由著名OpenResty社區成員Agent Zhang(章亦春)創建的。這是一個與OpenResty集成的Lua模塊,允許你在Nginx環境中直接進行Redis操作。利用OpenResty的強大功能,lua-resty-redis提供了異步非阻塞的Redis API,幫助開發者構建高性能、高并發的應用。
lua-resty-redis的核心在于其非阻塞I/O模型。它基于OpenResty的ngx.socket.tcp()接口實現,利用了LuaJIT的高效性能和Nginx事件循環機制。這意味著在處理大量并發請求時,即使Redis服務器繁忙,也不會導致Nginx的工作線程被阻塞,從而提高了整體系統的響應速度。
此外,該庫提供了豐富的Redis命令支持,包括但不限于數據讀寫、哈希操作、集合操作、有序集合、發布訂閱等。它的API清晰簡潔,易于理解和使用,使得你可以快速地將Redis功能融入到你的OpenResty應用程序中。
## 應用場景
* 緩存管理:利用Redis的高速讀寫能力,可以作為動態內容的高速緩存層,降低對后端數據庫的壓力。
* 分布式會話:借助lua-resty-redis,可以在多臺服務器之間共享用戶的會話狀態。
* 消息隊列:通過其發布的訂閱功能,可以構建簡單的消息隊列系統,實現任務的異步處理。
* 實時數據分析:利用Redis的數據結構,如計數器、集合等,進行實時統計和分析。
## 主要特點
* 非阻塞I/O - 基于OpenResty的異步TCP套接字接口,確保高并發場景下的效率。
* Redis命令支持 - 提供了Redis的所有主要命令,方便你執行各種操作。
* 錯誤處理 - 提供了良好的錯誤處理機制,便于定位和解決問題。
* 可擴展性 - 可以自定義連接池策略,適應不同規模和需求的系統。
* 簡潔API - 易于理解和使用的API,加速開發過程。
## 安裝
這里通過OPM工具包安裝,更多請查看[OpenResty實戰系列 | 包管理工具OPM和LuaRocks](https://mp.weixin.qq.com/s/s6-fukvBWdMuYRDa-cYZHw)
```lua
opm get openresty/lua-resty-redis
```
版本信息
```lua
# opm info openresty/lua-resty-redis
Name : lua-resty-redis
Version : 0.27
Abstract : Lua redis client driver for the ngx_lua based on the cosocket API
Author : Yichun "agentzh" Zhang (agentzh)
Account : openresty
Code Repo : https://github.com/openresty/lua-resty-redis
License : BSD 2-Clause "Simplified" or "FreeBSD" license
Original Work : yes
```
## 基礎使用
使用 lua-resty-redis 設置和獲取一個鍵值對基本示例
`redis_test_01.lua`文件
```lua
--[[-----------------------------------------------------------------------
* | Copyright (C) Shaobo Wan (Tinywan)
* |------------------------------------------------------------------------
--]]
-- redis config
local redis = require "resty.redis"
local red = redis:new()
red:set_timeouts(1000, 1000, 1000) -- 1 秒
-- 通過宿主機鏈接
local ok, err = red:connect("192.168.13.168", 6379)
if not ok then
ngx.say("[x] failed to connect: ", err)
return
end
-- 設置權限
local res, err = red:auth("123456")
if not res then
ngx.say("[x] failed to authenticate: ", err)
return
end
-- 設置權限
-- 請注意這里 auth 的調用過程
local count, err = red:get_reused_times()
ngx.say("[x] get_reused_times count: ", count)
if 0 == count then
res, err = red:auth("123456")
if not res then
ngx.say("[x]failed to authenticate: ", err)
return
end
elseif err then
ngx.say("[x]failed to get reused times: ", err)
return
end
ngx.say("[x] set result: ", ok)
local res, err = red:get("name")
if not res then
ngx.say("[x] failed to get name: ", err)
return
end
if res == ngx.null then
ngx.say("[x] name not found.")
return
end
ngx.say("[x] get name : ", res)
-- 連接池大小是100個,并且設置最大的空閑時間是 10 秒
local ok, err = red:set_keepalive(10000, 100)
if not ok then
ngx.say("[x] failed to set keepalive: ", err)
return
end
```
> **`red:get_reused_times()`方法**
* 如果當前連接不是從內建連接池中獲取的,該方法總是返回 `0` ,也就是說,該連接還沒有被使用過。
* 如果連接來自連接池,那么返回值永遠都是非零。
這個方法可以用來確認當前連接是否來自池子。對于 Redis 授權,實際上只需要建立連接后,首次認證一下,后面只需直接使用即可。換句話說,從連接池中獲取的連接都是經過授權認證的,只有新創建的連接才需要進行授權認證。所以大家就看到了 `count, err = red:get\_reused\_times()` 這段代碼,并有了下面 `if 0 == count then` 的判斷邏輯。
> 通過curl腳本測試請求打印結果
```lua
$ curl -i http://openresty.tinywan.com/lua_redis_test
HTTP/1.1 200 OK
Server: openresty/1.17.8.2
Date: Tue, 23 Jul 2024 07:16:23 GMT
Content-Type: text/html; charset=utf-8
Transfer-Encoding: chunked
Connection: keep-alive
Vary: Accept-Encoding
[x] set result: OK
[x] get name : Tinywan
```
如果指定的密碼是錯誤的,那么上面的示例將向HTTP客戶端輸出以下內容:
```
failed to authenticate: ERR invalid password
```
## 事務支持
這個庫支持Redis事務
`redis_transactions_test.lua` 文件
```lua
--[[-----------------------------------------------------------------------
* | Copyright (C) Shaobo Wan (Tinywan)
* |------------------------------------------------------------------------
--]]
local cjson = require "cjson"
-- redis config
local redis = require "resty.redis"
local red = redis:new()
red:set_timeouts(1000, 1000, 1000) -- 1 秒
-- 通過宿主機鏈接
local ok, err = red:connect("192.168.13.168", 6379)
if not ok then
ngx.say("[x] failed to connect: ", err)
return
end
-- 設置權限
local res, err = red:auth("123456")
if not res then
ngx.say("[x] failed to authenticate: ", err)
return
end
-- 設置權限
local ok, err = red:multi()
if not ok then
ngx.say("failed to run multi: ", err)
return
end
ngx.say("[x] multi ans: ", cjson.encode(ok))
local ans, err = red:set("resty_name", "Tinywan")
if not ans then
ngx.say("[x] failed to run sort: ", err)
return
end
ngx.say("[x] set ans: ", cjson.encode(ans))
local ans, err = red:lpop("resty_name")
if not ans then
ngx.say("[x] failed to run sort: ", err)
return
end
ngx.say("[x] set ans: ", cjson.encode(ans))
ans, err = red:exec()
ngx.say("[x] exec ans: ", cjson.encode(ans))
red:close()
```
通過curl腳本測試請求打印結果
```lua
$ curl -i http://openresty.tinywan.com/lua_redis_transactions_test
HTTP/1.1 200 OK
Server: openresty/1.17.8.2
Date: Tue, 23 Jul 2024 07:29:53 GMT
Content-Type: text/html; charset=utf-8
Transfer-Encoding: chunked
Connection: keep-alive
Vary: Accept-Encoding
[x] multi ans: "OK"
[x] set ans: "QUEUED"
[x] set ans: "QUEUED"
[x] exec ans: ["OK",[false,"WRONGTYPE Operation against a key holding the wrong kind of value"]]
```
## 小結
lua-resty-redis是一個強大的工具,能夠幫助開發者充分利用OpenResty和Redis的優勢,構建高性能的Web服務。無論是簡單的緩存解決方案還是復雜的分布式系統,它都能提供穩定且高效的支撐。如果你正在尋找一個在OpenResty環境中操作Redis的解決方案,那么lua-resty-redis絕對值得一試!
- 設計模式系列
- 工廠方法模式
- 序言
- Windows程序注冊為服務的工具WinSW
- 基礎
- 安裝
- 開發規范
- 目錄結構
- 配置
- 快速入門
- 架構
- 請求流程
- 架構總覽
- URL訪問
- 容器和依賴注入
- 中間件
- 事件
- 代碼層結構
- 四個層次
- 路由
- 控制器
- 請求
- 響應
- 數據庫
- MySQL實時同步數據到ES解決方案
- 阿里云DTS數據MySQL同步至Elasticsearch實戰
- PHP中的MySQL連接池
- PHP異步非阻塞MySQL客戶端連接池
- 模型
- 視圖
- 注解
- @SpringBootApplication(exclude={DataSourceAutoConfiguration.calss})
- @EnableFeignClients(basePackages = "com.wotu.feign")
- @EnableAspectJAutoProxy
- @EnableDiscoveryClient
- 錯誤和日志
- 異常處理
- 日志處理
- 調試
- 驗證
- 驗證器
- 驗證規則
- 擴展庫
- 附錄
- Spring框架知識體系詳解
- Maven
- Maven和Composer
- 構建Maven項目
- 實操課程
- 01.初識SpringBoot
- 第1章 Java Web發展史與學習Java的方法
- 第2章 環境與常見問題踩坑
- 第3章 springboot的路由與控制器
- 02.Java編程思想深度理論知識
- 第1章 Java編程思想總體
- 第2章 英雄聯盟的小案例理解Java中最為抽象的概念
- 第3章 徹底理解IOC、DI與DIP
- 03.Spring與SpringBoot理論篇
- 第1章 Spring與SpringBoot導學
- 第2章 Spring IOC的核心機制:實例化與注入
- 第3章 SpringBoot基本配置原理
- 04.SprinBoot的條件注解與配置
- 第1章 conditonal 條件注解
- 第2章 SpringBoot自動裝配解析
- 05.Java異常深度剖析
- 第1章 Java異常分類剖析與自定義異常
- 第2章 自動配置Url前綴
- 06.參數校驗機制與LomBok工具集的使用
- 第1章 LomBok工具集的使用
- 第2章 參數校驗機制以及自定義校驗
- 07.項目分層設計與JPA技術
- 第1章 項目分層原則與層與層的松耦合原則
- 第2章 數據庫設計、實體關系與查詢方案探討
- 第3章 JPA的關聯關系與規則查詢
- 08.ORM的概念與思維
- 第1章 ORM的概念與思維
- 第2章 Banner等相關業務
- 第3章 再談數據庫設計技巧與VO層對象的技巧
- 09.JPA的多種查詢規則
- 第1章 DozerBeanMapper的使用
- 第2章 詳解SKU的規格設計
- 第3章 通用泛型Converter
- 10.令牌與權限
- 第1章 通用泛型類與java泛型的思考
- 常見問題
- 微服務
- demo
- PHP中Self、Static和parent的區別
- Swoole-Cli
- 為什么要使用現代化PHP框架?
- 公眾號
- 一鍵部署微信公眾號Markdown編輯器(支持適配和主題設計)
- Autodesigner 2.0發布
- Luya 一個現代化PHP開發框架
- PHPZip - 創建、讀取和管理 ZIP 文件的簡單庫
- 吊打Golang的PHP界天花板webman壓測對比
- 簡潔而強大的 YAML 解析庫
- 推薦一個革命性的PHP測試框架:Kahlan
- ServBay下一代Web開發環境
- 基于Websocket和Canvas實現多人協作實時共享白板
- Apipost預執行腳本如何調用外部PHP語言
- 認證和授權的安全令牌 Bearer Token
- Laradock PHP 的 Docker 完整本地開發環境
- 高效接口防抖策略,確保數據安全,避免重復提交的終極解決方案!
- TIOBE 6月榜單:PHP穩步前行,編程語言生態的微妙變化
- Aho-Corasick字符串匹配算法的實現
- Redis鍵空間通知 Keyspace Notification 事件訂閱
- ServBay如何啟用并運行Webman項目
- 使用mpdf實現導出pdf文件功能
- Medoo 輕量級PHP數據庫框架
- 在PHP中編寫和運行單元測試
- 9 PHP運行時基準性能測試
- QR碼生成器在PHP中的源代碼
- 使用Gogs極易搭建的自助Git服務
- Gitea
- webman如何記錄SQL到日志?
- Sentry PHP: 實時監測并處理PHP應用程序中的錯誤
- Swoole v6 Alpha 版本已發布
- Proxypin
- Rust實現的Redis內存數據庫發布
- PHP 8.4.0 Alpha 1 測試版本發布
- 121
- Golang + Vue 開發的開源輕量 Linux 服務器運維管理面板
- 內網穿透 FRP VS Tailscale
- 新一代開源代碼托管平臺Gitea
- 微服務系列
- Nacos云原生配置中心介紹與使用
- 輕量級的開源高性能事件庫libevent
- 國密算法
- 國密算法(商用密碼)
- GmSSL 支持國密SM2/SM3/SM4/SM9/SSL 密碼工具箱
- GmSSL PHP 使用
- 數據庫
- SQLite數據庫的Web管理工具
- 阿里巴巴MySQL數據庫強制規范
- PHP
- PHP安全測試秘密武器 PHPGGC
- 使用declare(strict_types=1)來獲得更健壯的PHP代碼
- PHP中的魔術常量
- OSS 直傳阿里騰訊示例
- PHP源碼編譯安裝APCu擴展實現數據緩存
- BI性能DuckDB數據管理系統
- 為什么別人可以是架構師!而我卻不是?
- 密碼還在用 MD5 加鹽?不如試試 password_hash
- Elasticsearch 在電商領域的應用與實踐
- Cron 定時任務入門
- 如何動態設置定時任務!而不是寫死在Linux Crontab
- Elasticsearch的四種查詢方式,你知道多少?
- Meilisearch vs Elasticsearch
- OpenSearch vs Elasticsearch
- Emlog 輕量級開源博客及建站系統
- 現代化PHP原生協程引擎 PRipple
- 使用Zephir編寫C擴展將PHP源代碼編譯加密
- 如何將PHP源代碼編譯加密,同時保證代碼能正常的運行
- 為什么選擇Zephir給PHP編寫動態擴展庫?
- 使用 PHP + XlsWriter實現百萬級數據導入導出
- Rust編寫PHP擴展
- 阿里云盤開放平臺對接進行文件同步
- 如何構建自己的PHP靜態可執行文件
- IM后端架構
- RESTful設計方法和規范
- PHP編譯器BPC 7.3 發布,成功編譯ThinkPHP8
- 高性能的配置管理擴展 Yaconf
- PHP實現雪花算法庫 Snowflake
- PHP官方現代化核心加密庫Sodium
- pie
- 現代化、精簡、非阻塞PHP標準庫PSL
- PHP泛型和集合
- 手把手教你正確使用 Composer包管理
- JWT雙令牌認證實現無感Token自動續期
- 最先進PHP大模型深度學習庫TransformersPHP
- PHP如何啟用 FFI 擴展
- PHP超集語言PXP
- 低延遲雙向實時事件通信 Socket.IO
- PHP OOP中的繼承和多態
- 強大的現代PHP高級調試工具Kint
- PHP基金會
- 基于webman+vue3高質量中后臺框架SaiAdmin
- 開源免費的定時任務管理系統:Gocron
- 簡單強大OCR工具EasyOCR在PHP中使用
- PHP代碼抽象語法樹工具PHP AST Viewer
- MySQL數據庫管理工具PHPMyAdmin
- Rust編寫的一款高性能多人代碼編輯器Zed
- 超高性能PHP框架Workerman v5.0.0-beta.8 發布
- 高并發系列
- 入門介紹及安裝
- Lua腳本開發 Hello World
- 執行流程與階段詳解
- Nginx Lua API 接口開發
- Lua模塊開發
- OpenResty 高性能的正式原因
- 記一次查找 lua-resty-mysql 庫 insert_id 的 bug
- 包管理工具OPM和LuaRocks使用
- 異步非阻塞HTTP客戶端庫 lua-resty-http
- Nginx 內置綁定變量
- Redis協程網絡庫 lua-resty-redis
- 動態HTML渲染庫 lua-testy-template
- 單獨的
- StackBlitz在線開發環境
- AI
- 基礎概念
- 12312
- 基礎鏡像的坑
- 利用phpy實現 PHP 編寫 Vision Transformer (ViT) 模型
- 語義化版本 2.0.0