佳木斯湛栽影视文化发展公司

主頁 > 知識庫 > Redis Scan命令的基本使用方法

Redis Scan命令的基本使用方法

熱門標(biāo)簽:Linux服務(wù)器 地方門戶網(wǎng)站 鐵路電話系統(tǒng) 百度競價排名 服務(wù)外包 AI電銷 網(wǎng)站排名優(yōu)化 呼叫中心市場需求

1. 概述

SCAN 命令以及比較相近的 SSCAN、HSCAN 和 ZSCAN 命令都用于增量迭代數(shù)據(jù)集元素:

  • SCAN 命令用于迭代當(dāng)前數(shù)據(jù)庫中的數(shù)據(jù)庫鍵。
  • SSCAN 命令用于迭代集合(Set)中的元素。
  • HSCAN 命令用于迭代哈希(Hash)中的字段以及對應(yīng)的值。
  • ZSCAN 命令用于迭代有序集合(Sorted Set)中的元素以及對應(yīng)的得分。

由于這些命令都可以增量迭代,每次調(diào)用都只會返回少量元素,所以這些命令可以用于生產(chǎn)環(huán)境中,不用擔(dān)心像使用 KEYS、SMEMBERS 命令帶來的問題。在鍵或元素的大數(shù)據(jù)集上調(diào)用這些命令可能會長時間(甚至幾秒鐘)阻塞服務(wù)器。像 SMEMBERS 這樣的阻塞命令能夠在給定的時間內(nèi)提供數(shù)據(jù)集中所有的元素,但 SCAN 系列命令僅對返回的元素提供有限的保證,因為數(shù)據(jù)集在我們增量迭代時可能會發(fā)生改變。

SCAN,SSCAN,HSCAN 以及 ZSCAN 命令工作原理都非常類似,因此這篇文章會涵蓋這四個命令。區(qū)別在于 SSCAN,HSCAN 以及 ZSCAN 命令,第一個參數(shù)是保存 Set,Hash或 Sorted Set 值的鍵的名稱。SCAN命令不需要任何鍵名參數(shù),因為它會迭代當(dāng)前數(shù)據(jù)庫中所有的鍵,因此迭代的對象是數(shù)據(jù)庫本身。

2. 基本用法

SCAN 是基于游標(biāo)的迭代器。這意味著在每次調(diào)用該命令時,服務(wù)器都會返回一個更新后的新游標(biāo),用戶需要在下一次調(diào)用中將這個新游標(biāo)作為 SCAN 命令的游標(biāo)參數(shù)。當(dāng) SCAN 命令的游標(biāo)參數(shù)被設(shè)置為 0 時, 服務(wù)器將開始一次新的迭代,而當(dāng)服務(wù)器向用戶返回的新游標(biāo)為 0 時會終止迭代。以下是 SCAN 迭代的示例:

redis 127.0.0.1:6379> scan 0
1) "17"
2) 1) "key:12"
 2) "key:8"
 3) "key:4"
 4) "key:14"
 5) "key:16"
 6) "key:17"
 7) "key:15"
 8) "key:10"
 9) "key:3"
 10) "key:7"
 11) "key:1"
redis 127.0.0.1:6379> scan 17
1) "0"
2) 1) "key:5"
 2) "key:18"
 3) "key:0"
 4) "key:2"
 5) "key:19"
 6) "key:13"
 7) "key:6"
 8) "key:9"
 9) "key:11"

在上面的示例中,第一次調(diào)用使用 0 作為游標(biāo)來開始一次新的迭代。第二次調(diào)用時使用上一次調(diào)用返回的游標(biāo),即命令回復(fù)的第一個元素值,即17。從上面的示例可以看到,SCAN 命令返回值是兩個值的數(shù)組:第一個值是下一次調(diào)用中將要使用的新游標(biāo),第二個值是包含返回元素的數(shù)組。

由于在第二次調(diào)用中返回的游標(biāo)為 0,因此服務(wù)器向調(diào)用者發(fā)送信號,告知迭代已完成,并且遍歷完數(shù)據(jù)集。從游標(biāo)值 0 開始迭代,然后調(diào)用 SCAN 直到返回的游標(biāo)再次為 0,表示一個完整迭代。

3. 保證

SCAN 命令,以及其他增量迭代命令,在整個完整迭代過程中可以為用戶提供一系列的保證:

  • 在完整迭代開始直到完整迭代結(jié)束期間內(nèi)的所有元素都會被遍歷返回;這意味著,如果某個給定元素在開始迭代時位于數(shù)據(jù)集內(nèi),并且在終止迭代時仍然存在,那么 SCAN 會在某次迭代時返回給用戶。
  • 在完整迭代開始直到完整迭代結(jié)束期間內(nèi)不存在的元素永遠(yuǎn)都不會被返回;因此,如果某個元素在迭代開始之前就被刪除,并且在后續(xù)的迭代過程中從未添加回數(shù)據(jù)集中,那么 SCAN 永遠(yuǎn)都不會返回該元素 。

但是,由于 SCAN 只有很少的關(guān)聯(lián)狀態(tài)(僅有游標(biāo)),因此具有以下缺點:

  • 同一個元素可能會被返回多次。重復(fù)元素的問題需要我們自己的應(yīng)用程序處理, 例如,可以考慮將迭代返回的元素用于冪等操作(可以重復(fù)執(zhí)行多次操作)上。
  • 如果一個元素是在迭代過程中被添加到數(shù)據(jù)集的,又或者是在迭代過程中從數(shù)據(jù)集中被刪除的,那么這個元素可能會被返回,也可能不會。

4. 每次執(zhí)行返回數(shù)量

SCAN 系列的函數(shù)不能保證每次調(diào)用返回的元素數(shù)量會在給定范圍內(nèi)。每次調(diào)用可能會返回 0 個元素,但只要返回的游標(biāo)不為 0,客戶端就認(rèn)為迭代沒有結(jié)束(即使返回了 0 個元素也不能表示迭代的結(jié)束)。返回的元素數(shù)量會符合一定的規(guī)則:

  • 在迭代大型數(shù)據(jù)集時,SCAN 最多可能會返回幾十個元素。
  • 在迭代小的數(shù)據(jù)集并且內(nèi)部為編碼數(shù)據(jù)結(jié)構(gòu)時(小的 Set、Hashe 以及 Sorted Set),單次調(diào)用就可以返回數(shù)據(jù)集的所有元素。

但是,用戶可以使用 COUNT 參數(shù)來調(diào)整每次調(diào)用返回的元素的數(shù)量級。

5. COUNT參數(shù)

雖然 SCAN 不能保證每次迭代返回的元素數(shù)量,但是可以使用 COUNT 參數(shù)根據(jù)經(jīng)驗進(jìn)行調(diào)整?;旧希珻OUNT 參數(shù)的作用就是讓用戶告知迭代命令,在每次迭代中應(yīng)該從數(shù)據(jù)集里返回多少元素。雖然 COUNT 參數(shù)只是迭代命令實現(xiàn)上的一種提示(hint),但是在大多數(shù)情況下,這種提示是能滿足我們的預(yù)期:

  • COUNT 默認(rèn)值為 10。
  • 在迭代一個足夠大、由哈希表實現(xiàn)的數(shù)據(jù)庫、Set、Hash 或者 Sorted Set 時,如果用戶沒有使用 MATCH 參數(shù),那么每次調(diào)用返回 COUNT 個元素,或者比 COUNT 稍多的元素。
  • 在迭代一個編碼為 IntSet (一個只由整數(shù)值構(gòu)成的小數(shù)據(jù)集) 或 Hash 的 Set 以及編碼為 ZipList (由不同值構(gòu)成的小的 Hash 或者 Set) 的 Sorted Set 時,通常會無視 COUNT 參數(shù)指定的值,并在第一次調(diào)用時就將數(shù)據(jù)集包含的所有元素都返回給用戶。

沒有必要每次迭代都要使用相同的 COUNT 值。用戶可以在每次迭代中按自己的需要隨意改變 COUNT 值,只要記得將上次迭代返回的游標(biāo)用到下次迭代里面就可以了。

6. MATCH參數(shù)

我們也可以通過匹配一個 Glob 風(fēng)格的模式來迭代元素,類似于 KEYS 命令。我們只需要在 SCAN 命令后面追加 MATCH pattern> 參數(shù)即可實現(xiàn)。

以下是一個使用 MATCH 參數(shù)進(jìn)行迭代的示例:

redis 127.0.0.1:6379> sadd myset 1 2 3 foo foobar feelsgood
(integer) 6
redis 127.0.0.1:6379> sscan myset 0 match f*
1) "0"
2) 1) "foo"
 2) "feelsgood"
 3) "foobar"
redis 127.0.0.1:6379>

我們需要注意的是 MATCH 過濾器是在從數(shù)據(jù)集中檢索出元素之后,在將數(shù)據(jù)返回給客戶端之前應(yīng)用的。這意味著,如果模式匹配到數(shù)據(jù)集中很少的元素,則 SCAN 命令在很多次迭代中可能不返回元素。一個例子如下所示:

redis 127.0.0.1:6379> scan 0 MATCH *11*
1) "288"
2) 1) "key:911"
redis 127.0.0.1:6379> scan 288 MATCH *11*
1) "224"
2) (empty list or set)
redis 127.0.0.1:6379> scan 224 MATCH *11*
1) "80"
2) (empty list or set)
redis 127.0.0.1:6379> scan 80 MATCH *11*
1) "176"
2) (empty list or set)
redis 127.0.0.1:6379> scan 176 MATCH *11* COUNT 1000
1) "0"
2) 1) "key:611"
 2) "key:711"
 3) "key:118"
 4) "key:117"
 5) "key:311"
 6) "key:112"
 7) "key:111"
 8) "key:110"
 9) "key:113"
 10) "key:211"
 11) "key:411"
 12) "key:115"
 13) "key:116"
 14) "key:114"
 15) "key:119"
 16) "key:811"
 17) "key:511"
 18) "key:11"
redis 127.0.0.1:6379>

如上述所述,大多數(shù)調(diào)用沒有返回元素,而最后一次調(diào)用使用 COUNT 為1000,強制命令對該迭代進(jìn)行更多掃描,從而使得命令返回的元素也變多了。

7. TYPE參數(shù)

從 6.0 版開始,我們可以使用此參數(shù)要求 SCAN 命令僅返回與給定類型匹配的對象,從而允許我們遍歷數(shù)據(jù)庫以查找特定類型的鍵。SCAN 可以使用 TYPE 參數(shù),但 HSCAN 或 ZSCAN 等不可用。

type 參數(shù)與 TYPE 命令返回的字符串名稱相同。需要我們注意的是某些 Redis 類型(例如GeoHashes、HyperLogLogs、Bitmap 以及 Bitfields 等)其內(nèi)部是使用其他 Redis 類型(例如 String 或 Zset)來實現(xiàn)的,因此 SCAN 命令無法將其與相同類型的其他鍵區(qū)分開。例如,ZSET 和 GEOHASH:

redis 127.0.0.1:6379> GEOADD geokey 0 0 value
(integer) 1
redis 127.0.0.1:6379> ZADD zkey 1000 value
(integer) 1
redis 127.0.0.1:6379> TYPE geokey
zset
redis 127.0.0.1:6379> TYPE zkey
zset
redis 127.0.0.1:6379> SCAN 0 TYPE zset
1) "0"
2) 1) "geokey"
 2) "zkey"

重要的是,TYPE 過濾器是在從數(shù)據(jù)庫中檢索元素之后應(yīng)用的,因此該參數(shù)不會降低服務(wù)器完成完整迭代所需的負(fù)載,對于稀有類型,我們可能不會收到任何元素。

8. 多次并行迭代

不同客戶端可能在同一時間迭代同一數(shù)據(jù)集,客戶端每次執(zhí)行迭代都需要傳入一個游標(biāo),并在迭代結(jié)束之后獲得一個新的游標(biāo),而這個游標(biāo)就包含了迭代的所有狀態(tài),因此,服務(wù)器無須為迭代記錄任何狀態(tài)。

9. 在中間終止迭代

由于服務(wù)器端不會記錄狀態(tài),迭代的所有狀態(tài)都保存在游標(biāo)中,因此調(diào)用方可以自由地中途終止迭代,不用向服務(wù)器發(fā)送通知。An infinite number of iterations can be started and never terminated without any issue.

10. 使用錯誤的游標(biāo)調(diào)用SCAN

使用錯誤的,負(fù)數(shù)的,超出范圍的游標(biāo)或其他無效的游標(biāo)來調(diào)用 SCAN,會導(dǎo)致未定義的行為,但絕不會導(dǎo)致崩潰。未定義的是指 SCAN 將不再確保返回元素的保證。

唯一有效的游標(biāo)是:

開始迭代時的游標(biāo)值為0。

上一次調(diào)用 SCAN 返回的游標(biāo),以便繼續(xù)迭代。

11. 終止保證

只有在保證迭代的數(shù)據(jù)集大小始終保持在給定的最大上限內(nèi)時(大小恒定),才能保證 SCAN 算法能終止;否則,對一直增長的數(shù)據(jù)集進(jìn)行迭代可能會導(dǎo)致 SCAN 永遠(yuǎn)不會終止迭代(死循環(huán))。

這很容易直觀地看出:如果數(shù)據(jù)集不斷增長,為了訪問所有可能出現(xiàn)的元素,將需要做越來越多的工作,而能否結(jié)束一個迭代取決于對 SCAN 的調(diào)用次數(shù)、COUNT 參數(shù)值以及數(shù)據(jù)集的增長速度。

12. 返回值

SCAN,SSCAN,HSCAN 以及 ZSCAN 命令都返回一個包含兩個元素的回復(fù),第一個元素表示游標(biāo)的無符號64位整數(shù),第二個元素是迭代出的元素數(shù)組:

SCAN 元素數(shù)組是鍵的列表。

SSCAN 元素數(shù)組是 Set 成員的列表。

HSCAN 元素數(shù)組包含兩個元素,即字段和值,對應(yīng) Hash 的每個返回元素。

ZSCAN 元素數(shù)組包含兩個元素,即一個成員及其關(guān)聯(lián)的分?jǐn)?shù),對應(yīng) Sorted Set 中的每個返回元素。

總結(jié)

以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,謝謝大家對腳本之家的支持。

您可能感興趣的文章:
  • Redis中scan命令的深入講解
  • php redis擴展支持scan命令實現(xiàn)方法
  • Redis中Scan命令的基本使用教程
  • 詳解Redis SCAN命令實現(xiàn)有限保證的原理
  • Redis中Scan命令的踩坑實錄
  • redis中scan命令的基本實現(xiàn)方法

標(biāo)簽:湖南 銅川 蘭州 衡水 崇左 湘潭 仙桃 黃山

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《Redis Scan命令的基本使用方法》,本文關(guān)鍵詞  ;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問題,煩請?zhí)峁┫嚓P(guān)信息告之我們,我們將及時溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無關(guān)。
  • 相關(guān)文章
  • 收縮
    • 微信客服
    • 微信二維碼
    • 電話咨詢

    • 400-1100-266
    正蓝旗| 寻甸| 陆河县| 漳平市| 洞头县| 门头沟区| 霸州市| 连州市| 扎赉特旗| 岚皋县| 中山市| 虹口区| 文昌市| 乌拉特前旗| 高阳县| 北碚区| 西青区| 东乡县| 阿拉善左旗| 韶关市| 柯坪县| 公主岭市| 榆林市| 广平县| 仙居县| 伊金霍洛旗| 舞钢市| 苏尼特左旗| 二连浩特市| 吕梁市| 大名县| 同德县| 常宁市| 磴口县| 曲麻莱县| 华亭县| 南城县| 阿克苏市| 临海市| 邹城市| 唐山市|