C++中的备忘录模式

目录

[备忘录模式(Memento Pattern)](#备忘录模式(Memento Pattern))

实际应用

文本编辑器的撤销功能

游戏角色状态保存和恢复

图形编辑器的撤销/重做功能

总结


备忘录模式(Memento Pattern)

备忘录模式是一种行为型设计模式,它允许在不破坏封装性的前提下捕获和恢复对象的内部状态。这个模式的主要目的是保存对象的某个状态,以便在适当的时候恢复对象到之前的状态。备忘录模式主要包含三个角色:

  1. Originator :负责创建一个备忘录(Memento ),用以记录当前的内部状态,并在需要时使用备忘录恢复内部状态。

  2. Memento :备忘录,负责存储Originator 的内部状态。

  3. Caretaker:负责保存备忘录,但不能对备忘录的内容进行操作或检查。

实际应用

文本编辑器的撤销功能

-- 文本编辑器,支持撤销操作。

cpp 复制代码
#include <iostream>
#include <vector>
#include <string>

// Memento类,用于存储文本编辑器的状态
class Memento {
private:
    std::string state;

public:
    Memento(const std::string& state) : state(state) {}
    std::string getState() const { return state; }
};

// Originator类,表示文本编辑器
class TextEditor {
private:
    std::string text;

public:
    void write(const std::string& words) {
        text += words;
    }

    std::string getText() const {
        return text;
    }

    Memento save() const {
        return Memento(text);
    }

    void restore(const Memento& memento) {
        text = memento.getState();
    }
};

// Caretaker类,用于管理Memento
class Caretaker {
private:
    std::vector<Memento> history;

public:
    void save(const Memento& memento) {
        history.push_back(memento);
    }

    Memento undo() {
        if (!history.empty()) {
            Memento memento = history.back();
            history.pop_back();
            return memento;
        }
        return Memento(""); // 返回空的Memento
    }
};

// 客户端代码
int main() {
    TextEditor editor;
    Caretaker caretaker;

    editor.write("Hello ");
    caretaker.save(editor.save());

    editor.write("World!");
    caretaker.save(editor.save());

    std::cout << "Current Text: " << editor.getText() << std::endl;

    editor.restore(caretaker.undo());
    std::cout << "After Undo: " << editor.getText() << std::endl;

    editor.restore(caretaker.undo());
    std::cout << "After Undo: " << editor.getText() << std::endl;

    return 0;
}

游戏角色状态保存和恢复

-- 模拟一个游戏角色的状态保存和恢复功能。

cpp 复制代码
#include <iostream>
#include <vector>
#include <string>

// Memento类,用于存储游戏角色的状态
class Memento {
private:
    int health;
    int mana;
    int experience;

public:
    Memento(int health, int mana, int experience)
        : health(health), mana(mana), experience(experience) {}

    int getHealth() const { return health; }
    int getMana() const { return mana; }
    int getExperience() const { return experience; }
};

// Originator类,表示游戏角色
class GameCharacter {
private:
    int health;
    int mana;
    int experience;

public:
    GameCharacter(int health, int mana, int experience)
        : health(health), mana(mana), experience(experience) {}

    void setState(int health, int mana, int experience) {
        this->health = health;
        this->mana = mana;
        this->experience = experience;
    }

    void displayState() const {
        std::cout << "Health: " << health << ", Mana: " << mana
                  << ", Experience: " << experience << std::endl;
    }

    Memento save() const {
        return Memento(health, mana, experience);
    }

    void restore(const Memento& memento) {
        health = memento.getHealth();
        mana = memento.getMana();
        experience = memento.getExperience();
    }
};

// Caretaker类,用于管理Memento
class Caretaker {
private:
    std::vector<Memento> savepoints;

public:
    void save(const Memento& memento) {
        savepoints.push_back(memento);
    }

    Memento load() {
        if (!savepoints.empty()) {
            Memento memento = savepoints.back();
            savepoints.pop_back();
            return memento;
        }
        return Memento(100, 100, 0); // 返回初始状态
    }
};

// 客户端代码
int main() {
    GameCharacter character(100, 50, 0);
    Caretaker caretaker;

    character.displayState();
    caretaker.save(character.save());

    character.setState(80, 40, 10);
    character.displayState();
    caretaker.save(character.save());

    character.setState(50, 30, 20);
    character.displayState();

    character.restore(caretaker.load());
    character.displayState();

    character.restore(caretaker.load());
    character.displayState();

    return 0;
}

图形编辑器的撤销/重做功能

-- 图形编辑器,支持撤销和重做操作。

cpp 复制代码
#include <iostream>
#include <vector>
#include <string>
#include <stack>

// Memento类,用于存储图形编辑器的状态
class Memento {
private:
    std::vector<std::string> shapes;

public:
    Memento(const std::vector<std::string>& shapes) : shapes(shapes) {}
    std::vector<std::string> getState() const { return shapes; }
};

// Originator类,表示图形编辑器
class GraphicEditor {
private:
    std::vector<std::string> shapes;

public:
    void addShape(const std::string& shape) {
        shapes.push_back(shape);
    }

    void displayShapes() const {
        for (const auto& shape : shapes) {
            std::cout << shape << " ";
        }
        std::cout << std::endl;
    }

    Memento save() const {
        return Memento(shapes);
    }

    void restore(const Memento& memento) {
        shapes = memento.getState();
    }
};

// Caretaker类,用于管理Memento
class Caretaker {
private:
    std::stack<Memento> undoStack;
    std::stack<Memento> redoStack;

public:
    void save(const Memento& memento) {
        undoStack.push(memento);
        while (!redoStack.empty()) {
            redoStack.pop();
        }
    }

    Memento undo() {
        if (!undoStack.empty()) {
            Memento memento = undoStack.top();
            undoStack.pop();
            redoStack.push(memento);
            return memento;
        }
        return Memento(std::vector<std::string>()); // 返回空的Memento
    }

    Memento redo() {
        if (!redoStack.empty()) {
            Memento memento = redoStack.top();
            redoStack.pop();
            undoStack.push(memento);
            return memento;
        }
        return Memento(std::vector<std::string>()); // 返回空的Memento
    }
};

// 客户端代码
int main() {
    GraphicEditor editor;
    Caretaker caretaker;

    editor.addShape("Circle");
    caretaker.save(editor.save());

    editor.addShape("Square");
    caretaker.save(editor.save());

    editor.addShape("Triangle");
    caretaker.save(editor.save());

    std::cout << "Current Shapes: ";
    editor.displayShapes();

    editor.restore(caretaker.undo());
    std::cout << "After Undo: ";
    editor.displayShapes();

    editor.restore(caretaker.undo());
    std::cout << "After Undo: ";
    editor.displayShapes();

    editor.restore(caretaker.redo());
    std::cout << "After Redo: ";
    editor.displayShapes();

    editor.restore(caretaker.redo());
    std::cout << "After Redo: ";
    editor.displayShapes();

    return 0;
}

总结

备忘录模式帮助我们在不破坏对象封装性的前提下捕获和恢复对象的内部状态,从而实现撤销和重做操作。

相关推荐
Ni-Guvara几秒前
函数对象笔记
c++·算法
似霰5 分钟前
安卓智能指针sp、wp、RefBase浅析
android·c++·binder
芊寻(嵌入式)15 分钟前
C转C++学习笔记--基础知识摘录总结
开发语言·c++·笔记·学习
獨枭17 分钟前
C++ 项目中使用 .dll 和 .def 文件的操作指南
c++
霁月风19 分钟前
设计模式——观察者模式
c++·观察者模式·设计模式
橘色的喵20 分钟前
C++编程:避免因编译优化引发的多线程死锁问题
c++·多线程·memory·死锁·内存屏障·内存栅栏·memory barrier
一颗松鼠24 分钟前
JavaScript 闭包是什么?简单到看完就理解!
开发语言·前端·javascript·ecmascript
有梦想的咸鱼_26 分钟前
go实现并发安全hashtable 拉链法
开发语言·golang·哈希算法
海阔天空_201331 分钟前
Python pyautogui库:自动化操作的强大工具
运维·开发语言·python·青少年编程·自动化
天下皆白_唯我独黑38 分钟前
php 使用qrcode制作二维码图片
开发语言·php