Redis鍵空間通知 Keyspace Notification 事件訂閱
## 概述
本文所說的定時任務或者說計劃任務并不是很多人想象中的那樣,比如說每天凌晨三點自動運行起來跑一個腳本。這種都已經爛大街了,隨便一個 Crontab 就能搞定了。
這里所說的定時任務可以說是計時器任務,比如說用戶觸發了某個動作,那么從這個點開始過二十四小時我們要對這個動作做點什么。那么如果有 1000 個用戶觸發了這個動作,就會有 1000 個定時任務。于是這就不是 Cron 范疇里面的內容了。
? ? ? ?舉個最簡單的例子,一個用戶推薦了另一個用戶,我們定一個二十四小時之后的任務,看看被推薦的用戶有沒有來注冊,如果沒注冊就給他搞一條短信過去。
## 功能概覽
鍵空間通知使得客戶端可以通過訂閱頻道或模式, 來接收那些以某種方式改動了 Redis 數據集的事件。事件通過 Redis 的訂閱與發布功能(pub/sub)來進行分發, 因此所有支持訂閱與發布功能的客戶端都可以在無須做任何修改的情況下, 直接使用鍵空間通知功能。
## 鍵空間消息
在 Redis 的 2.8.0 版本之后,其推出了一個新的特性——鍵空間消息(Redis Keyspace Notifications),它配合 2.0.0 版本之后的 SUBSCRIBE 就能完成這個定時任務的操作了,不過定時的單位是秒。
#### Publish / Subscribe?
Redis 在 2.0.0 之后推出了 Pub / Sub 的指令,大致就是說一邊給 Redis 的特定頻道發送消息,另一邊從 Redis 的特定頻道取值——形成了一個簡易的消息隊列。
#### Redis Keyspace Notifications
在 Redis 里面有一些事件,比如鍵到期、鍵被刪除等。然后我們可以通過配置一些東西來讓 Redis 一旦觸發這些事件的時候就往特定的 Channel 推一條消息。
大致的流程就是我們給 Redis 的某一個 db 設置過期事件,使其鍵一旦過期就會往特定頻道推消息,我在自己的客戶端這邊就一直消費這個頻道就好了。
以后一來一條定時任務,我們就把這個任務狀態壓縮成一個鍵,并且過期時間為距這個任務執行的時間差。那么當鍵一旦到期,就到了任務該執行的時間,Redis 自然會把過期消息推去,我們的客戶端就能接收到了。這樣一來就起到了定時任務的作用。
## 配置
因為開啟鍵空間通知功能需要消耗一些 CPU , 所以在默認配置下, 該功能處于關閉狀態。可以通過修改?`redis.conf`?文件, 或者直接使用?CONFIG?SET?命令來開啟或關閉鍵空間通知功能。
* 當?`notify-keyspace-events?`選項的參數為空字符串時,功能關閉。
* 當參數不是空字符串時,功能開啟。
> `notify-keyspace-events`?的參數可以是以下字符的任意組合, 它指定了服務器該發送哪些類型的通知

輸入的參數中至少要有一個?K?或者?E?, 否則的話, 不管其余的參數是什么, 都不會有任何通知被分發。舉個例子, 如果只想訂閱鍵空間中和列表相關的通知, 那么參數就應該設為?Kl?, 諸如此類。
#### 具體配置
首先找到redis.conf配置文件,打開文件,查找`notify-keyspace-events`,將前面的`#`去掉即可。

>注意:這里配置的是`notify-keyspace-events`的`Ex`參數,即說明,當鍵過期的時候會觸發通知,如果只需要哈希命令鍵觸發通知則可以設置為`notify-keyspace-events Eh`。
配置完成重啟`redis-server`即可
## 使用
#### 命令行
開啟一個終端,redis-cli 進入 redis 。開始訂閱所有操作,等待接收消息。
```
127.0.0.1:6379> psubscribe __keyevent@0__:expired
Reading messages... (press Ctrl-C to quit)
1) "psubscribe"
2) "__keyevent@0__:expired"
3) (integer) 1
```
再開啟一個終端,redis-cli 進入 redis,新增一個 20秒過期的鍵
```
127.0.0.1:6379> SETEX username 30 Tinywan
OK
127.0.0.1:6379> get username
"Tinywan"
127.0.0.1:6379> TTL username
(integer) 25
127.0.0.1:6379>
```
另外一邊執行了阻塞訂閱操作后的終端,20秒過期后有如下信息輸出:
```
127.0.0.1:6379> psubscribe __keyevent@0__:expired
Reading messages... (press Ctrl-C to quit)
1) "psubscribe"
2) "__keyevent@0__:expired"
3) (integer) 1
1) "pmessage"
2) "__keyevent@0__:expired"
3) "__keyevent@0__:expired"
4) "username"
```
輸出以上信息,說明對過期Key信息的訂閱是成功的。
發布訂閱截圖

### 代碼
Redis實例類RedisInstance
```php
<?php
/**
* @desc RedisInstance.php 描述信息
* @author Tinywan(ShaoBo Wan)
* @date 2024/6/26 21:36
*/
declare(strict_types=1);
class RedisInstance
{
private Redis $redis;
/**
* @param string $host
* @param int $port
* @throws RedisException
*/
public function __construct(string $host = '127.0.0.1', int $port = 6379)
{
$this->redis = new Redis();
$this->redis->connect($host, $port);
}
/**
* @desc expire
* @param null $key
* @param int $time
* @return bool|Redis
* @throws RedisException
* @author Tinywan(ShaoBo Wan)
*/
public function expire($key = null, int $time = 0)
{
return $this->redis->expire($key, $time);
}
/**
* @desc psubscribe
* @param $callback
* @param array $patterns
* @throws RedisException
* @author Tinywan(ShaoBo Wan)
*/
public function psubscribe($callback, array $patterns = [])
{
$this->redis->psubscribe($patterns, $callback);
}
/**
* @desc setOption
* @author Tinywan(ShaoBo Wan)
*/
public function setOption()
{
$this->redis->setOption(\Redis::OPT_READ_TIMEOUT, -1);
}
}
```
訂閱文件`psubscribe.php`
```
<?php
/**
* @desc psubscribe.php 描述信息
* @author Tinywan(ShaoBo Wan)
* @date 2024/6/26 21:39
*/
require '../vendor/autoload.php';
$redis = new \RedisInstance('dnmp-redis');
$redis->setOption();
$redis->psubscribe(function ($redis, $pattern, $channel, $msg){
echo 'Pattern:' . $pattern .PHP_EOL;
echo 'Channel:' . $channel .PHP_EOL;
echo 'Message:' . $msg .PHP_EOL;
}, ['__keyevent@0__:expired']);
```
運行`psubscribe.php` 觀察訂閱狀態
```
# php psubscribe.php
Pattern:__keyevent@0__:expired
Channel:__keyevent@0__:expired
Message:username
Pattern:__keyevent@0__:expired
Channel:__keyevent@0__:expired
Message:username2
Pattern:__keyevent@0__:expired
Channel:__keyevent@0__:expired
Message:username3
```
發布事件
```
127.0.0.1:6379> SETEX username 3 Tinywan
OK
127.0.0.1:6379> SETEX username2 5 Tinywan
OK
127.0.0.1:6379> SETEX username3 5 Tinywan
```
## 小結結
通過以上步驟,成功地實現了Redis鍵空間通知使用。首先配置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