一、概述
在軟件開發中,有時需要保存一個對象的狀態,以便於允許用戶取消相關操作或者從以往的狀態中恢復過來。比如一個文檔版本管理系統,可以根據需要將指定文檔恢復到之前保存過的任意一個狀態。這時就可以通過備忘錄模式來實現。
二、備忘錄模式
備忘錄模式可以在不破壞封裝性的前提下捕獲一個對象的內部狀態,並在該對象之外保存這個狀態。其結構圖如下:
Memento用於保存Originator對象的內部狀態。
Originator創建Memento,並根據需要決定需要在Memento中保存那些狀態,同時還能從Memento中恢復內部狀態。
Caretaker負責保存Memento對象,但不對Memento對象進行任何操作。
備忘錄模式用來保存與對象有關的數據,這樣可以在將來對對象進行復原。例如在繪圖程序中需要保存對象的顏色尺寸等。
三、示例
我們我們就以文檔版本管理系統為例來介紹備忘錄模式。
首先定義Originator,在這裡就是文檔本身。
? 1 2 3 4 5 6 7 8 9 10 11 12 13 14public
class
Document
{
public
string
Content {
get
;
set
; }
public
DocumentVersion CreateMemento()
{
return
new
DocumentVersion( Content);
}
public
void
SetMemento(DocumentVersion documentVersion)
{
Content = documentVersion.Content;
}
}
接著定義Memento。
? 1 2 3 4 5 6 7 8 9public
class
DocumentVersion
{
public
string
Content {
get
;
set
; }
public
DocumentVersion(
string
content)
{
Content = content;
}
}
最後定義Caretaker,通過一個Dictionary容器保存Memento。
? 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15public
class
Caretaker
{
private
Dictionary<
int
, DocumentVersion> _mementoList =
new
Dictionary<
int
, DocumentVersion>();
public
DocumentVersion GetDocumentVersion(
int
versionID)
{
return
_mementoList[versionID];
}
public
void
AddDocumentVersion(DocumentVersion documentVersion)
{
int
maxVersionID = _mementoList.Keys.Count == 0 ? 0: _mementoList.Keys.Max();
_mementoList.Add(maxVersionID + 1, documentVersion);
}
}
四、備忘錄模式的適用場景
在以下情況下可以考慮使用備忘錄模式:
如果系統需要提供回滾操作時,使用備忘錄模式非常合適。例如文本編輯器的Ctrl+Z撤銷操作的實現,數據庫中事務操作。
五、備忘錄模式的優缺點
備忘錄模式具有以下優點:
如果某個操作錯誤地破壞了數據的完整性,此時可以使用備忘錄模式將數據恢復成原來正確的數據。
備份的狀態數據保存在發起人角色之外,這樣發起人就不需要對各個備份的狀態進行管理。而是由備忘錄角色進行管理,而備忘錄角色又是由管理者角色管理,符合單一職責原則。
當然,備忘錄模式也存在一定的缺點:
在實際的系統中,可能需要維護多個備份,需要額外的資源,這樣對資源的消耗比較嚴重。
六、總結
備忘錄模式主要思想是——利用備忘錄對象來對保存發起人的內部狀態,當發起人需要恢復原來狀態時,再從備忘錄對象中進行獲取,在實際開發過程也應用到這點,例如數據庫中的事務處理。