轉載:解密Redis持久化

来源:http://www.cnblogs.com/heartinharbin/archive/2017/06/16/7026581.html
-Advertisement-
Play Games

本文內容來源於Redis作者博文,Redis作者說,他看到的所有針對Redis的討論中,對Redis持久化的誤解是最大的,於是他寫了一篇長文來對Redis的持久化進行了系統性的論述。文章非常長,也很值得一看,NoSQLFan將主要內容簡述成本文。 什麼是持久化,簡單來講就是將數據放到斷電後數據不會丟 ...


本文內容來源於Redis作者博文,Redis作者說,他看到的所有針對Redis的討論中,對Redis持久化的誤解是最大的,於是他寫了一篇長文來對Redis的持久化進行了系統性的論述。文章非常長,也很值得一看,NoSQLFan將主要內容簡述成本文。

什麼是持久化,簡單來講就是將數據放到斷電後數據不會丟失的設備中。也就是我們通常理解的硬碟上。

寫操作的流程

首先我們來看一下資料庫在進行寫操作時到底做了哪些事,主要有下麵五個過程。

  1. 客戶端向服務端發送寫操作(數據在客戶端的記憶體中)
  2. 資料庫服務端接收到寫請求的數據(數據在服務端的記憶體中)
  3. 服務端調用write(2) 這個系統調用,將數據往磁碟上寫(數據在系統記憶體的緩衝區中)
  4. 操作系統將緩衝區中的數據轉移到磁碟控制器上(數據在磁碟緩存中)
  5. 磁碟控制器將數據寫到磁碟的物理介質中(數據真正落到磁碟上)

故障分析

寫操作大致有上面5個流程,下麵我們結合上面的5個流程看一下各種級別的故障。

  • 當資料庫系統故障時,這時候系統內核還是OK的,那麼此時只要我們執行完了第3步,那麼數據就是安全的,因為後續操作系統會來完成後面幾步,保證數據最終會落到磁碟上。
  • 當系統斷電,這時候上面5項中提到的所有緩存都會失效,並且資料庫和操作系統都會停止工作。所以只有當數據在完成第5步後,機器斷電才能保證數據不丟失,在上述四步中的數據都會丟失。

通過上面5步的瞭解,可能我們會希望搞清下麵一些問題:

  • 資料庫多長時間調用一次write(2),將數據寫到內核緩衝區
  • 內核多長時間會將系統緩衝區中的數據寫到磁碟控制器
  • 磁碟控制器又在什麼時候把緩存中的數據寫到物理介質上

對於第一個問題,通常資料庫層面會進行全面控制。而對第二個問題,操作系統有其預設的策略,但是我們也可以通過POSIX API提供的fsync系列命令強制操作系統將數據從內核區寫到磁碟控制器上。對於第三個問題,好像資料庫已經無法觸及,但實際上,大多數情況下磁碟緩存是被設置關閉的。或者是只開啟為讀緩存,也就是寫操作不會進行緩存,直接寫到磁碟。建議的做法是僅僅當你的磁碟設備有備用電池時才開啟寫緩存。

數據損壞

所謂數據損壞,就是數據無法恢復,上面我們講的都是如何保證數據是確實寫到磁碟上去,但是寫到磁碟上可能並不意味著數據不會損壞。比如我們可能一次寫請求會進行兩次不同的寫操作,當意外發生時,可能會導致一次寫操作安全完成,但是另一次還沒有進行。如果資料庫的數據文件結構組織不合理,可能就會導致數據完全不能恢復的狀況出現。

這裡通常也有三種策略來組織數據,以防止數據文件損壞到無法恢復的情況:

  • 第一種是最粗糙的處理,就是不通過數據的組織形式保證數據的可恢復性。而是通過配置數據同步備份的方式,在數據文件損壞後通過數據備份來進行恢復。實際上MongoDB在不開啟journaling日誌,通過配置Replica Sets時就是這種情況。
  • 另一種是在上面基礎上添加一個操作日誌,每次操作時記一下操作的行為,這樣我們可以通過操作日誌來進行數據恢復。因為操作日誌是順序追加的方式寫的,所以不會出現操作日誌也無法恢復的情況。這也類似於MongoDB開啟了journaling日誌的情況。
  • 更保險的做法是資料庫不進行老數據的修改,只是以追加方式去完成寫操作,這樣數據本身就是一份日誌,這樣就永遠不會出現數據無法恢復的情況了。實際上CouchDB就是此做法的優秀範例。

RDB快照

下麵我們說一下Redis的第一個持久化策略,RDB快照。Redis支持將當前數據的快照存成一個數據文件的持久化機制。而一個持續寫入的資料庫如何生成快照呢。Redis藉助了fork命令的copy on write機制。在生成快照時,將當前進程fork出一個子進程,然後在子進程中迴圈所有的數據,將數據寫成為RDB文件。

我們可以通過Redis的save指令來配置RDB快照生成的時機,比如你可以配置當10分鐘以內有100次寫入就生成快照,也可以配置當1小時內有1000次寫入就生成快照,也可以多個規則一起實施。這些規則的定義就在Redis的配置文件中,你也可以通過Redis的CONFIG SET命令在Redis運行時設置規則,不需要重啟Redis。

Redis的RDB文件不會壞掉,因為其寫操作是在一個新進程中進行的,當生成一個新的RDB文件時,Redis生成的子進程會先將數據寫到一個臨時文件中,然後通過原子性rename系統調用將臨時文件重命名為RDB文件,這樣在任何時候出現故障,Redis的RDB文件都總是可用的。

同時,Redis的RDB文件也是Redis主從同步內部實現中的一環。

但是,我們可以很明顯的看到,RDB有他的不足,就是一旦資料庫出現問題,那麼我們的RDB文件中保存的數據並不是全新的,從上次RDB文件生成到Redis停機這段時間的數據全部丟掉了。在某些業務下,這是可以忍受的,我們也推薦這些業務使用RDB的方式進行持久化,因為開啟RDB的代價並不高。但是對於另外一些對數據安全性要求極高的應用,無法容忍數據丟失的應用,RDB就無能為力了,所以Redis引入了另一個重要的持久化機制:AOF日誌。

AOF日誌

aof日誌的全稱是append only file,從名字上我們就能看出來,它是一個追加寫入的日誌文件。與一般資料庫的binlog不同的是,AOF文件是可識別的純文本,它的內容就是一個個的Redis標準命令。比如我們進行如下實驗,使用Redis2.6版本,在啟動命令參數中設置開啟aof功能:

./redis-server --appendonly yes

然後我們執行如下的命令:

redis 127.0.0.1:6379> set key1 Hello
OK
redis 127.0.0.1:6379> append key1 " World!"
(integer) 12
redis 127.0.0.1:6379> del key1
(integer) 1
redis 127.0.0.1:6379> del non_existing_key
(integer) 0

這時我們查看AOF日誌文件,就會得到如下內容:

$ cat appendonly.aof
*2
$6
SELECT
$1
0
*3
$3
set
$4
key1
$5
Hello
*3
$6
append
$4
key1
$7
 World!
*2
$3
del
$4
key1

 

可以看到,寫操作都生成了一條相應的命令作為日誌。其中值得註意的是最後一個del命令,它並沒有被記錄在AOF日誌中,這是因為Redis判斷出這個命令不會對當前數據集做出修改。所以不需要記錄這個無用的寫命令。另外AOF日誌也不是完全按客戶端的請求來生成日誌的,比如命令INCRBYFLOAT在記AOF日誌時就被記成一條SET記錄,因為浮點數操作可能在不同的系統上會不同,所以為了避免同一份日誌在不同的系統上生成不同的數據集,所以這裡只將操作後的結果通過SET來記錄。

AOF重寫

你可以會想,每一條寫命令都生成一條日誌,那麼AOF文件是不是會很大?答案是肯定的,AOF文件會越來越大,所以Redis又提供了一個功能,叫做AOF rewrite。其功能就是重新生成一份AOF文件,新的AOF文件中一條記錄的操作只會有一次,而不像一份老文件那樣,可能記錄了對同一個值的多次操作。其生成過程和RDB類似,也是fork一個進程,直接遍曆數據,寫入新的AOF臨時文件。在寫入新文件的過程中,所有的寫操作日誌還是會寫到原來老的AOF文件中,同時還會記錄在記憶體緩衝區中。當重完操作完成後,會將所有緩衝區中的日誌一次性寫入到臨時文件中。然後調用原子性的rename命令用新的AOF文件取代老的AOF文件。

從上面的流程我們能夠看到,RDB和AOF操作都是順序IO操作,性能都很高。而同時在通過RDB文件或者AOF日誌進行資料庫恢復的時候,也是順序的讀取數據載入到記憶體中。所以也不會造成磁碟的隨機讀。

AOF可靠性設置

AOF是一個寫文件操作,其目的是將操作日誌寫到磁碟上,所以它也同樣會遇到我們上面說的寫操作的5個流程。那麼寫AOF的操作安全性又有多高呢。實際上這是可以設置的,在Redis中對AOF調用write(2)寫入後,何時再調用fsync將其寫到磁碟上,通過appendfsync選項來控制,下麵appendfsync的三個設置項,安全強度逐漸變強。

appendfsync no

當設置appendfsync為no的時候,Redis不會主動調用fsync去將AOF日誌內容同步到磁碟,所以這一切就完全依賴於操作系統的調試了。對大多數Linux操作系統,是每30秒進行一次fsync,將緩衝區中的數據寫到磁碟上。

appendfsync everysec

當設置appendfsync為everysec的時候,Redis會預設每隔一秒進行一次fsync調用,將緩衝區中的數據寫到磁碟。但是當這一次的fsync調用時長超過1秒時。Redis會採取延遲fsync的策略,再等一秒鐘。也就是在兩秒後再進行fsync,這一次的fsync就不管會執行多長時間都會進行。這時候由於在fsync時文件描述符會被阻塞,所以當前的寫操作就會阻塞。

所以,結論就是,在絕大多數情況下,Redis會每隔一秒進行一次fsync。在最壞的情況下,兩秒鐘會進行一次fsync操作。

這一操作在大多數資料庫系統中被稱為group commit,就是組合多次寫操作的數據,一次性將日誌寫到磁碟。

appednfsync always

當設置appendfsync為always時,每一次寫操作都會調用一次fsync,這時數據是最安全的,當然,由於每次都會執行fsync,所以其性能也會受到影響。

對於pipelining有什麼不同

對於pipelining的操作,其具體過程是客戶端一次性發送N個命令,然後等待這N個命令的返回結果被一起返回。通過採用pipilining就意味著放棄了對每一個命令的返回值確認。由於在這種情況下,N個命令是在同一個執行過程中執行的。所以當設置appendfsync為everysec時,可能會有一些偏差,因為這N個命令可能執行時間超過1秒甚至2秒。但是可以保證的是,最長時間不會超過這N個命令的執行時間和。

與postgreSQL和MySQL的比較

這一塊就不多說了,由於上面操作系統層面的數據安全已經講了很多,所以其實不同的資料庫在實現上都大同小異。總之最後的結論就是,在Redis開啟AOF的情況下,其單機數據安全性並不比這些成熟的SQL資料庫弱。

數據導入

這些持久化的數據有什麼用,當然是用於重啟後的數據恢復。Redis是一個記憶體資料庫,無論是RDB還是AOF,都只是其保證數據恢復的措施。所以Redis在利用RDB和AOF進行恢復的時候,都會讀取RDB或AOF文件,重新載入到記憶體中。相對於MySQL等資料庫的啟動時間來說,會長很多,因為MySQL本來是不需要將數據載入到記憶體中的。

但是相對來說,MySQL啟動後提供服務時,其被訪問的熱數據也會慢慢載入到記憶體中,通常我們稱之為預熱,而在預熱完成前,其性能都不會太高。而Redis的好處是一次性將數據載入到記憶體中,一次性預熱。這樣只要Redis啟動完成,那麼其提供服務的速度都是非常快的。

而在利用RDB和利用AOF啟動上,其啟動時間有一些差別。RDB的啟動時間會更短,原因有兩個,一是RDB文件中每一條數據只有一條記錄,不會像AOF日誌那樣可能有一條數據的多次操作記錄。所以每條數據只需要寫一次就行了。另一個原因是RDB文件的存儲格式和Redis數據在記憶體中的編碼格式是一致的,不需要再進行數據編碼工作。在CPU消耗上要遠小於AOF日誌的載入。

好了,大概內容就說到這裡。更詳細完整的版本請看Redis作者的博文:Redis persistence demystified。本文如有描述不周之處,就大家指正。

 

轉載出處:http://blog.nosqlfan.com/html/3813.html


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 最近做項目要求使用到網路,想來想去選擇了AsyncHttpClient框架開進行APP開發。在這裡把我工作期間遇到的問題以及對AsyncHttpClient的使用經驗做出相應總結,希望能對您的學習有所幫助。 首先按照慣例先來簡單瞭解一些AsyncHttpClient網路框架的一些知識。 1、簡介 A ...
  • description 在開發過程中, 往往會有很多的model來裝載屬性. 而在開發期間經常會進行調試查看model里的屬性值是否正確. 那麼問題來了, 在objective-c里使用NSLog("%@",model)這行代碼列印出來的卻是model的地址. 不是我們所想要的結果~! 看圖: 那麼 ...
  • 登錄的業務邏輯{ http:是短連接. 伺服器如何判斷當前用戶是否登錄? // 1. 如果是即時通信類:長連接. // 如何保證伺服器跟客戶端保持長連接狀態? // "心跳包" 用來檢測用戶是否線上!用來做長連接! http:短連接使用token 機制來驗證用戶安全性 // token 值: 登錄令 ...
  • 有些時候我們使用Service的時需要採用隱私啟動的方式,但是Android 5.0一齣來後,其中有個特性就是Service Intent must be explitict,也就是說從Lollipop開始,service服務必須採用顯示方式啟動。 而android源碼是這樣寫的(源碼位置:sdk/ ...
  • 轉自 http://www.jianshu.com/p/6588c69b42cf ...
  • 基於AS的採用Kotlin語言開發的動畫漸入的弧形菜單(附java代碼),具體看效果... ...
  • NSCharacterSet 對於string處理,在IOS生態系統中經常用到且用錯的重要組成部分:NSCharacterSet。下麵詳解一下: NSCharacterSet ,以及它的可變版本NSMutableCharacterSet,用面向對象的方式來表示一組Unicode字元。它經常與NSSt ...
  • google 在Android 5.0推出 Camera2 這個類,用於替換 Camera,但是Camera2要求android sdk 最低版本為 minSdkVersion = 21 (5.0系統),所以Camera2 還不能完全替換 Camera,在相容低版本的時候,還是需要兩者一起協同開發。 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...