摘要:首發(fā)于樊浩柏科學院的管道是用來打包多條無關命令批量執(zhí)行,以減少多個命令分別執(zhí)行帶來的網(wǎng)絡交互時間。在一些批量操作數(shù)據(jù)的場景,使用管道可以顯著提升的讀寫性能。這時,可以使用管道實現(xiàn),也可以內(nèi)嵌腳本實現(xiàn)。
首發(fā)于 樊浩柏科學院
Redis 的 管道 (pipelining)是用來打包多條無關命令批量執(zhí)行,以減少多個命令分別執(zhí)行帶來的網(wǎng)絡交互時間。在一些批量操作數(shù)據(jù)的場景,使用管道可以顯著提升 Redis 的讀寫性能。
原理演示Redis 的管道實質就是命令打包批量執(zhí)行,多次網(wǎng)絡交互減少到單次。使用管道和不使用管道時的交互過程如下:
我們使用 nc 命令來直觀感受下 Redis 管道的使用過程:
# 安裝nc命令 $ yum install nc # nc打包多個命令 $ (printf "PING PING PING ") | nc localhost 6379 # 響應 +PONG +PONG +PONG
因此,只要通過管道進行命令打包后,Redis 就可以批量返回命令的執(zhí)行結果了。
管道的應用首先,構造示例需要的 Hash 用戶數(shù)據(jù):
$keyPrex = "user:hash:u:"; for ($i=1; $i<=10000; $i++) { $redis->hMset($keyPrex.$i, [ "name" => name(), //name()函數(shù)生成隨機姓名 "age" => rand(21, 30), "sex" => rand(0, 1), "is_new" => rand(0, 1) ]); }
然后,查看導入 Redis 中的數(shù)據(jù):
127.0.0.1:6379> keys user:hash:u:* 9997) "user:hash:u:3013" 9998) "user:hash:u:8971" 9999) "user:hash:u:4761" 10000) "user:hash:u:1828" 127.0.0.1:6379> HGETALL user:hash:u:1828 1) "name" 2) "ggrg" 3) "age" 4) "23" 5) "sex" 6) "0" 7) "is_new" 8) "1"需求
在某個社交活動中,通過一系列篩選邏輯后取得種子用戶 uid,然后用這些 uid 去 Hash 獲取用戶的信息。這種情況下你會怎么來處理呢?
不使用管道一般情況下,在數(shù)據(jù)量較小時,我們會直接使用 HGETALL 命令遍歷地獲取用戶數(shù)據(jù)。
$start = nowTime(); foreach (range(1, 1000) as $id) { $user[] = $redis->hgetAll($keyPrex.$id); } echo "時間:", nowTime() - $start, "ms", PHP_EOL; 時間:39ms
執(zhí)行所用時間:39ms
使用管道因為通過 uid 批量獲取用戶數(shù)據(jù),各個命令并沒有依賴關系,所以可以使用 Redis 的管道來優(yōu)化查詢。
$start = nowTime(); $redis->multi(Redis::PIPELINE); foreach (range(1, 1000) as $id) { //返回資源id相同的socket資源,并未執(zhí)行命令 $redis->hgetAll($keyPrex.$id); } $user = $redis->exec(); echo "時間:", nowTime() - $start, "ms", PHP_EOL; 時間:6ms
使用管道后,執(zhí)行時間顯著地減少為:6ms。使用 tcpdump 抓取打包后的命令如下:
10:45:03.029049 IP localhost.58176 > localhost.6379: Flags [P.], seq 2255478840:2255479211, ack 3144685411, win 342, options [nop,nop,TS val 17640474 ecr 17640474], length 371 E..../@[email protected][email protected]....... ,.*2 $7 HGETALL $13 user:hash:u:1 *2 $7 HGETALL $13 user:hash:u:2 *2 $7 ... ...適用場景
在批量操作(查詢和寫入)數(shù)據(jù)時,我們應盡量避免多次跟 Redis 的網(wǎng)絡交互。這時,可以使用管道實現(xiàn),也可以 Redis 內(nèi)嵌 Lua 腳本實現(xiàn)。需要注意的是:
管道只適用于無因果關聯(lián)的多命令操作,否則就需要借助 Lua 腳本實現(xiàn)批量操作;
在實際應用中,Redis 往往不可能是單機部署,如果想要在集群中使用管道,可以部署為一主多從架構,此時所有節(jié)點的數(shù)據(jù)都一致,隨機選取節(jié)點使用管道即可;
總結在批量獲取數(shù)據(jù)時,盡管使用 Redis 的管道性能會顯著提升,但是使用管道時 Redis 會緩存之前命令的結果,最后一并輸出給終端,因此所打包的命令不宜太多,否則內(nèi)存使用會很嚴重。
文章版權歸作者所有,未經(jīng)允許請勿轉載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉載請注明本文地址:http://systransis.cn/yun/29790.html
摘要:優(yōu)化的一些建議盡量使用短的當然在精簡的同時,不要完了的見名知意。的開發(fā)者向的開發(fā)者之一詢問優(yōu)化方案,得到的回復是使用結構。所以內(nèi)存分配策略應該設置為表示內(nèi)核允許分配所有的物理內(nèi)存,而不管當前的內(nèi)存狀態(tài)如何。 優(yōu)化的一些建議 1、盡量使用短的key 當然在精簡的同時,不要完了key的見名知意。對于value有些也可精簡,比如性別使用0、1。 2、避免使用keys * keys *, 這個...
閱讀 3664·2021-09-22 15:15
閱讀 3567·2021-08-12 13:24
閱讀 1314·2019-08-30 15:53
閱讀 1826·2019-08-30 15:43
閱讀 1189·2019-08-29 17:04
閱讀 2798·2019-08-29 15:08
閱讀 1586·2019-08-29 13:13
閱讀 3091·2019-08-29 11:06