23种设计模式(C++)之 备忘录(Memento)模式
意图
在不破坏封装性的前提下,捕获一个对象的内部状态,并在该对象之外保存这个状态。这样以后就可将对象恢复到原先保存的状态。
适用性
- 必须保存一个对象在某一个时刻的(部分)状态,这样以后需要时它才能恢复到先前的状态。
- 如果一个用接口来让其它对象直接得到这些状态,将会暴露对象的实现细节并破坏对象的封装性。
场景
比如说下棋时的悔棋功能,文档的撤销,回复撤销功能等等。
角色
- Mementor
- 备忘录存储原发器对象的内部状态。原发器根据需要决定备忘录存储原发器的哪些内部状态。
- 防止原发器以外的其他对象访问备忘录。备忘录实际上有两个接口,管理者(caretaker)只能看到备忘录的窄接口——它只能将备忘录传递给其他对象。相反,原发器能够看到一个宽接口,允许它访问返回到先前状态所需的所有数据。理想的情况是只允许生成本备忘录的那个原发器访问本备忘录的内部状态。
- Originator
- 原发器创建一个备忘录,用以记录当前时刻它的内部状态。
- 使用备忘录恢复内部状态。
- Caretaker
- 负责保存好备忘录。
- 不能对备忘录的内容进行操作或检查。
实例
- 创建备忘录类
class Memento
{
private:
string state;
public:
Memento(string state)
{
this->state = state;
}
string getState()
{
return this->state;
}
};
- 创建原发器类
class Originator
{
private:
string state;
public:
void writeState(string state)
{
this->state = state;
}
string getState()
{
return this->state;
}
void printState()
{
cout << "Current state is '" + this->state+"'" << endl;
}
Memento* saveState2Memento()
{
return new Memento(this->state);
}
void restoreFromMemento(Memento* memento)
{
this->state = memento->getState();
}
};
- 创建负责人类
class CareTaker
{
private:
vector<Memento*> memento_vec;
public:
void addMemento2List(Memento* memento)
{
this->memento_vec.push_back(memento);
}
Memento* getMementoByIndex(size_t index)
{
return this->memento_vec.at(index);
}
};
- Client
int main()
{
//Create a caretaker
CareTaker* careTaker = new CareTaker();
//Create an originator
Originator* originator = new Originator();
//write and print state
originator->writeState("First step");
originator->printState();
originator->writeState("Second step");
originator->printState();
//save state to memento
careTaker->addMemento2List(originator->saveState2Memento());
originator->writeState("Third step");
originator->printState();
careTaker->addMemento2List(originator->saveState2Memento());
//restore state from the list saved by caretaker
originator->restoreFromMemento(careTaker->getMementoByIndex(0));
originator->printState();
}
- 结果
Current state is 'First step'
Current state is 'Second step'
Current state is 'Third step'
Current state is 'Second step'