這篇文章主要介紹“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命令詳解”的學習就結束了,希望能夠解決大家的疑惑。理論與實踐的搭配能更好的幫助大家學習,快去試試吧!若想繼續學習更多相關知識,請繼續關注創新互聯網站,小編會繼續努力為大家帶來更多實用的文章!
網頁名稱:redis4.0下MEMORY命令詳解
標題來源:http://vcdvsql.cn/article38/phogsp.html
成都網站建設公司_創新互聯,為您提供服務器托管、網站設計公司、網站收錄、外貿網站建設、、移動網站建設
聲明:本網站發布的內容(圖片、視頻和文字)以用戶投稿、用戶轉載內容為主,如果涉及侵權請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網站立場,如需處理請聯系客服。電話:028-86922220;郵箱:631063699@qq.com。內容未經允許不得轉載,或轉載時需注明來源: 創新互聯