溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

redis4.0下MEMORY命令詳解

發布時間:2021-08-31 02:17:15 來源:億速云 閱讀:208 作者:chen 欄目:關系型數據庫

這篇文章主要介紹“redis4.0下MEMORY命令詳解”,在日常操作中,相信很多人在redis4.0下MEMORY命令詳解問題上存在疑惑,小編查閱了各式資料,整理出簡單好用的操作方法,希望對大家解答”redis4.0下MEMORY命令詳解”的疑惑有所幫助!接下來,請跟著小編一起來學習吧!

前言

在過去,查看redis的內存使用狀態只有info memory命令,而且也只有一些基礎信息,想要獲取全局信息就有些困難。4.0開始redis提供了MEMORY命令,一切都變得簡單起來。

MEMORY命令

MEMORY命令一共有5個子命令,可以通過MEMORY HELP來查看:

127.0.0.1:6379> memory help

1) "MEMORY DOCTOR - Outputs memory problems report"

2) "MEMORY USAGE <key> [SAMPLES <count>] - Estimate memory usage of key"

3) "MEMORY STATS - Show memory usage details"

4) "MEMORY PURGE - Ask the allocator to release memory"

5) "MEMORY MALLOC-STATS - Show allocator internal stats"

接下來我們從MEMORY STATS開始,一一介紹各個子命令的功能。

1. MEMORY STATS

首先,我們需要明確一個概念,redis的內存使用不僅包含所有的key-value數據,還有描述這些key-value的元信息,以及許多管理功能的消耗,比如持久化、主從復制,通過MEMORY STATS可以更好的了解到redis的內存使用狀況。

這里我們啟動了一個打開持久化功能并且帶slave的redis,向其中隨機寫入了一些數據(某些數據還帶有過期時間),以便讀者可以更好的了解redis的內存使用,接下來執行MEMORY STATS命令:

127.0.0.1:6379> memory stats

 1) "peak.allocated"

 2) (integer) 423995952

 3) "total.allocated"

 4) (integer) 11130320

 5) "startup.allocated"

 6) (integer) 9942928

 7) "replication.backlog"

 8) (integer) 1048576

 9) "clients.slaves"

10) (integer) 16858

11) "clients.normal"

12) (integer) 49630

13) "aof.buffer"

14) (integer) 3253

15) "db.0"

16) 1) "overhead.hashtable.main"

    2) (integer) 5808

    3) "overhead.hashtable.expires"

    4) (integer) 104

17) "overhead.total"

18) (integer) 11063904

19) "keys.count"

20) (integer) 94

21) "keys.bytes-per-key"

22) (integer) 12631

23) "dataset.bytes"

24) (integer) 66416

25) "dataset.percentage"

26) "5.5934348106384277"

27) "peak.percentage"

28) "2.6251003742218018"

29) "fragmentation"

30) "1.1039986610412598"

一共有15項內容,內存使用量均以字節為單位,我們一個一個來看:

1. peak.allocated

redis啟動到現在,最多使用過多少內存。

2. total.allocated

當前使用的內存總量。

3. startup.allocated

redis啟動初始化時使用的內存,有很多讀者會比較奇怪,為什么我的redis啟動以后什么都沒做就已經占用了幾十MB的內存?

這是因為redis本身不僅存儲key-value,還有其他的內存消耗,比如共享變量、主從復制、持久化和db元信息,下面各項會有詳細介紹。

4. replication.backlog

主從復制backlog使用的內存,默認10MB,backlog只在主從斷線重連時發揮作用,主從復制本身并不依賴此項。

5. clients.slaves

主從復制中所有slave的讀寫緩沖區,包括output-buffer(也即輸出緩沖區)使用的內存和querybuf(也即輸入緩沖區),這里簡單介紹一下主從復制:

redis把一次事件循環中,所有對數據庫發生更改的內容先追加到slave的output-buffer中,在事件循環結束后統一發送給slave。

那么主從之間就難免會有數據的延遲,如果主從之間連接斷開,重連時為了保證數據的一致性就要做一次全量同步,這顯然是不夠高效的。backlog就是為此而設計,master在backlog中緩存一部分主從復制的增量數據,斷線重連時如果slave的偏移量在backlog中,那就可以只把偏移量之后的增量數據同步給slave即可,避免了全量同步的開銷。

6. clients.normal

除slave外所有其他客戶端的讀寫緩沖區。

有時候一些客戶端讀取不及時,就會造成output-buffer積壓占用內存過多的情況,可以通過配置項client-output-buffer-limit來限制,當超過閾值之后redis就會主動斷開連接以釋放內存,slave亦是如此。

7. aof.buffer

此項為aof持久化使用的緩存和aofrewrite時產生的緩存之和,當然如果關閉了appendonly那這項就一直為0:

redis并不是在有寫入時就立即做持久化的,而是在一次事件循環內把所有的寫入數據緩存起來,待到事件循環結束后再持久化到磁盤。

aofrewrite時緩存增量數據使用的內存,只在aofrewrite時才會使用,aofrewrite機制可以參考之前的文章《redis4.0之利用管道優化aofrewrite》。

可以看出這一項的大小與寫入流量成正比。

8. db.0

redis每個db的元信息使用的內存,這里只使用了db0,所以只打印了db0的內存使用狀態,當使用其他db時也會有相應的信息。

db的元信息有以下三項:

a) redis的db就是一張hash表,首先就是這張hash表使用的內存(redis使用鏈式hash,hash表中存放所有鏈表的頭指針);

b) 每一個key-value對都有一個dictEntry來記錄他們的關系,元信息便包含該db中所有dictEntry使用的內存;

c) redis使用redisObject來描述value所對應的不同數據類型(string、list、hash、set、zset),那么redisObject占用的空間也計算在元信息中。

overhead.hashtable.main:

db的元信息也即是以上三項之和,計算公式為:

hashtable + dictEntry + redisObject

overhead.hashtable.expires:

對于key的過期時間,redis并沒有把它和value放在一起,而是單獨用一個hashtable來存儲,但是expires這張hash表記錄的是key-expire信息,所以不需要`redisObject`來描述value,其元信息也就少了一項,計算公式為:

hashtable + dictEntry

9. overhead.total

3-8項之和:startup.allocated+replication.backlog+clients.slaves+clients.normal+aof.buffer+dbx

10. dataset.bytes

所有數據所使用的內存——也即total.allocated - overhead.total——當前內存使用量減去管理類內存使用量。

11. dataset.percentage

所有數據占比,這里并沒有直接使用total.allocated做分母,而是除去了redis啟動初始化的內存,計算公式為:

100 * dataset.bytes / (total.allocated - startup.allocated)

12. keys.count

redis當前存儲的key總量

13. keys.bytes-per-key

平均每個key的內存大小,直覺上應該是用dataset.bytes除以keys.count即可,但是redis并沒有這么做,而是把管理類內存也平攤到了每個key的內存使用中,計算公式為:

(total.allocated - startup.allocated) / keys.count

14. peak.percentage

當前使用內存與歷史最高值比例

15. fragmentation

內存碎片率

2. MEMORY USAGE

相信所有redis用戶都希望對每一個key-value的內存使用了如指掌,然而4.0之前redis并沒有提供一個明確的方法來進行內存評估,不過從4.0開始,MEMORY命令實現了這一功能。

首先看下使用方法:MEMORY usage [samples]

命令參數不多,通過字面意思也可以看出來是評估指定key的內存使用情況。samples是可選參數默認為5,以hash為例看下其如果工作:

首先類似于上一節中的overhead.hashtable.main,要計算hash的元信息內存,包括hash表的大小以及所有dictEntry的內存占用信息。

與overhead.hashtable.main不同的是,每個dictEntry中key-value都是字符串,所以沒redisObject的額外消耗。在評估真正的數據內存大小時redis并沒有去遍歷所有key,而是采用的抽樣估算:隨機抽取samples個key-value對計算其平均內存占用,再乘以key-value對的個數即得到結果。試想一下如果要精確計算內存占用,那么就需要遍歷所有的元素,當元素很多時就是使redis阻塞,所以請合理設置samples的大小。

其他數據結構的計算方式類似于hash,此處就不再贅述。

3. MEMORY DOCTOR

此項子命令是作者給出的關于redis內存使用方面的建議,在不同的允許狀態下會有不同的分析結果:

首先是沒問題的情況

運行狀態良好:

Hi Sam, I can't find any memory issue in your instance. I can only account for what occurs on this base.

redis的數據量很小,暫無建議:

Hi Sam, this instance is empty or is using very little memory, my issues detector can't be used in these conditions. Please, leave for your mission on Earth and fill it with some data. The new Sam and I will be back to our programming as soon as I finished rebooting.

接下來出現的結果就需要注意了

內存使用峰值1.5倍于目前內存使用量,此時內存碎片率可能會比較高,需要注意:

Peak memory: In the past this instance used more than 150% the memory that is currently using. The allocator is normally not able to release memory after a peak, so you can expect to see a big fragmentation ratio, however this is actually harmless and is only due to the memory peak, and if the Redis instance Resident Set Size (RSS) is currently bigger than expected, the memory will be used as soon as you fill the Redis instance with more data. If the memory peak was only occasional and you want to try to reclaim memory, please try the MEMORY PURGE command, otherwise the only other option is to shutdown and restart the instance.

內存碎片率過高超過1.4,需要注意:

High fragmentation: This instance has a memory fragmentation greater than 1.4 (this means that the Resident Set Size of the Redis process is much larger than the sum of the logical allocations Redis performed). This problem is usually due either to a large peak memory (check if there is a peak memory entry above in the report) or may result from a workload that causes the allocator to fragment memory a lot. If the problem is a large peak memory, then there is no issue. Otherwise, make sure you are using the Jemalloc allocator and not the default libc malloc.

每個slave緩沖區的平均內存超過10MB,原因可能是master寫入流量過高,也有可能是主從同步的網絡帶寬不足或者slave處理較慢:

Big slave buffers: The slave output buffers in this instance are greater than 10MB for each slave (on average). This likely means that there is some slave instance that is struggling receiving data, either because it is too slow or because of networking issues. As a result, data piles on the master output buffers. Please try to identify what slave is not receiving data correctly and why. You can use the INFO output in order to check the slaves delays and the CLIENT LIST command to check the output buffers of each slave.

普通客戶端緩沖區的平均內存超過200KB,原因可能是pipeline使用不當或者Pub/Sub客戶端處理消息不及時導致:

Big client buffers: The clients output buffers in this instance are greater than 200K per client (on average). This may result from different causes, like Pub/Sub clients subscribed to channels bot not receiving data fast enough, so that data piles on the Redis instance output buffer, or clients sending commands with large replies or very large sequences of commands in the same pipeline. Please use the CLIENT LIST command in order to investigate the issue if it causes problems in your instance, or to understand better why certain clients are using a big amount of memory.

4. MEMORY MALLOC-STATS

打印內存分配器狀態,只在使用jemalloc時有用。

5. MEMORY PURGE

請求分配器釋放內存,同樣只對jemalloc生效。

到此,關于“redis4.0下MEMORY命令詳解”的學習就結束了,希望能夠解決大家的疑惑。理論與實踐的搭配能更好的幫助大家學習,快去試試吧!若想繼續學習更多相關知識,請繼續關注億速云網站,小編會繼續努力為大家帶來更多實用的文章!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

亚洲午夜精品一区二区_中文无码日韩欧免_久久香蕉精品视频_欧美主播一区二区三区美女