This is the 24th day of my participation in the August More Text Challenge
1. “Regret medicine” for program data
Memo mode is a design “moonlight box” that allows us to travel back in time; Is the program data “regret medicine”, eat it can return to a state; Is the designer’s “reassurance”, even if the object is damaged, but also can be restored to a relatively intact state.
Memo mode is very common, such as: document undo, database transactions can be used to achieve the memo mode, this article to the editor undo function as an example.
Suppose you now have an editor software that can edit titles and content, and present documents, with the code as simple as this:
class Editor {
String title;/ / title
String content;/ / content
/ / display
void show(a) {
print("{title='" + title +"', content='" + content +"'}"); }}Copy the code
Client call:
main() {
Editor editor = new Editor();
editor.setTitle("Title");
editor.setContent("Content..."); editor.show(); } output: {title='title', content='content... '}
Copy the code
Everything works fine, but the editor has a bug that doesn’t allow you to “undo,” so if you make a mistake, you can’t go back to your previous state. Human beings are fallible animals and it is impossible not to make mistakes, so we have to improve the software.
Each save of the document should create a copy that the user can undo at any time and restore to the previous state. How do you optimize it? Modify the Editor class? This violates the “on/off principle”. Is there a more elegant way to do this?
The answer, of course, is yes. Create a Memento memo class that creates a copy of the memo each time the user saves a document.
class Memento {
// Once a memo is created, it cannot be modified
final String title;
final String content;
}
Copy the code
The Editor class has been slightly modified to now support creating memo objects and restoring data from memos:
class Editor {
String title;/ / title
String content;/ / content
// Create a memo object
Memento createMemento(a){
return new Memento(title, content);
}
// Restore data from the memo
void restoreMemento(Memento memento) {
if(memento ! =null) {
this.title = memento.getTitle();
this.content = memento.getContent(); }}}Copy the code
Now there is a new problem, how to manage the created memo object? Give it to the Editor to manage? The Editor is only responsible for editing and displaying the document, and then imposing a memo management role on it violates the “single-responsibility rule”, so the best way is to create a memo management Caretaker.
class Caretaker {
Stack<Memento> stack = new Stack<>();
void save(Memento memento) {
stack.push(memento);
}
Memento restore(a){
if (stack.size() > 0) {
return stack.pop();
}
throw new Exception("Irrevocable..."); }}Copy the code
Memo I use a “stack” structure to manage, you think ah, every time save data is not on the stack, undo must be restored to the latest data, that is not out of the stack, so the “stack” structure is very suitable (do not consider here reverse undo).
Now that undo is done, the responsibilities of the class are clear. The class diagram looks like this:
Simple, this is the memo mode!
2. Definition of memo mode
Capture the internal state of an object and store the state outside of the object without breaking encapsulation. You can then restore the object to its original saved state.
Memorandum pattern generic class diagram
- Originator: Originator role, responsible for defining which properties need to be backed up, and for creating and restoring memo objects.
- Memento: A Memento role that stores the status of Originator and provides the status during restoration.
- -caretaker: Caretaker who manages and maintains memos.
The memo model is very simple, the responsibilities of each class are very clear, and it provides a way to fill the gaps in the real world and make “regret medicine” work in the world of applications.
3. Application of memo mode
The standard memo model is rare and has some variations in practice, but the main idea remains the same.
You can consider using memos whenever your class objects need to recover data or to revert to a previous state.
Use memo mode has some matters need to pay attention to: 1. Maintain the life cycle of the memo memo object will be created in “recently” is used, if you don’t use, remember to remove the reference, make its can be GC recycling, or create a lot of unused memo object, GC and cannot be recycled, can cause a memory leak.
Note that the performance memo object needs to back up the status attribute of the initiator, so it will occupy some memory space, especially if there are too many properties. Remember not to frequently create the memo object, and limit the maximum number of the memo object to avoid damaging the system unconsciously.
4. To summarize
Memo mode can restore application data to the past state, making “regret medicine” in the application world real and feasible. For example, Java developers can easily use the Clone function to realize the backup of objects, so that the initiator role can also assume the role of the memo, and the structure of the class will become simpler.