Make writing a habit together! This is the second day of my participation in the “Gold Digging Day New Plan · April More text challenge”. Click here for more details.
Model motivation
Suppose you had an object and wanted to generate an exact copy, how would you do it?
You probably create a new object of the same class, then iterate over all its member variables and copy the values of the member variables into the new object.
Yeah, that’s pretty straightforward. There is a problem, however, that not all objects can be copied directly in this way, because some objects are invisible, given that they have private member variables. In addition, you need to know the object’s class to create replicas, so your code must rely on that class.
Based on the above questions, the prototype pattern emerges, and its working mode is as follows:
The prototype pattern delegates the cloning process to the actual object being cloned. The pattern declares a common interface (in Java, the implementation Cloneable interface) for all objects that support cloning, which allows you to clone objects without coupling the code to the class that the object belongs to. Objects that support cloning are prototypes, which implement cloning methods that do not require you to create new objects.
Mitosis of cells is a good analogy for archetypal patterns
define
Prototype mode, also known as clone mode, belongs to the creation mode. It can copy an existing object without knowing any creation details, so that the code does not depend on the class of the object.
UML class diagrams
The model structure
The archetypal pattern contains the following characters:
Prototype
Abstract stereotype classes declare clone methods, and in most cases only one of them will be namedclone
The method ofConcretePrototype
: Concrete prototype classes will implement cloning methods (any class can beConcretePrototype
, just need to implementPrototype
Interface can be)Client
The client can copy any object that implements the prototype interface
Due to the special implementation in Java, which is somewhat different from the generic UML class diagram implementation above, I decided to explain the UML class diagram of the prototype pattern in Java and its implementation separately.
A UML class diagram of the Java stereotype pattern
public class Object {
// clone() is defined in Object class.
protected native Object clone(a) throws CloneNotSupportedException;
}
Copy the code
All Java classes inherit from java.lang.object, and the Object class itself provides a clone() method that makes a copy of a Java Object, so all classes implement the Clone () method by default. So the implementation of the prototype pattern in the Java language is simple.
A Java class that implements cloning must implement a Cloneable interface. This interface is null and does not declare any methods. Implements implements a class that supports replication. If a class does not implement the interface but call clone () method, the Java compiler throws a CloneNotSupportedException anomalies (such as a code block).
Tips: For any object obj, the clone() method satisfies:
obj.clone() ! = obj
That is, the cloned object and the original object are not the same objectobj.clone().getClass() == obj.getClass()
That is, the type of the cloned object is the same as that of the original object- If the object obj
equals()
Method is properly defined, thenobj.clone().equals(x)
Should returntrue
More instances
details about shallow (shallow copy) and deep (deep copy) cloning will be explained in the “schema extension” section at the end of this article!
Mail replication (shallow clone)
Because the email object contains the content more (such as sender, receiver, title, content, date, accessories, etc.), a system now need to provide a mail replication, for you have created a good email object, you can through the way of copying objects to create a new mail, if need to change some parts, without modifying the original email object, You only need to modify the copied mail object. Design the system using prototype patterns. In this example, shallow clone is used to implement mail replication, that is, mail Email is copied without the Attachment reference object.
Mail replication (deep clone)
The use of deep cloning to achieve mail replication, that is, copy the mail and also copy the attachment.
The sample code
Object.java
Core classes in the JDK
public class Object {
// clone() is defined in Object class.
protected native Object clone(a) throws CloneNotSupportedException;
}
Copy the code
Cloneable.java
Core classes in the JDK
package java.lang;
/**
* A class implements the <code>Cloneable</code> interface to
* indicate to the {@linkjava.lang.Object#clone()} method that it * is legal for that method to make a * field-for-field copy of instances of that class. * <p> * Invoking Object's clone method on an instance that does not implement the * <code>Cloneable</code> interface results in the exception * <code>CloneNotSupportedException</code> being thrown. * <p> * By convention, classes that implement this interface should override * <tt>Object.clone</tt> (which is protected) with a public method. * See {@link java.lang.Object#clone()} for details on overriding this
* method.
* <p>
* Note that this interface does <i>not</i> contain the <tt>clone</tt> method.
* Therefore, it is not possible to clone an object merely by virtue of the
* fact that it implements this interface. Even if the clone method is invoked
* reflectively, there is no guarantee that it will succeed.
*
* @author unascribed
* @see java.lang.CloneNotSupportedException
* @see java.lang.Object#clone()
* @sinceJDK1.0 * /
public interface Cloneable {}Copy the code
ConcretePrototype.java
public class ConcretePrototype implements Cloneable {
@Override
protected Object clone(a) throws CloneNotSupportedException {
return super.clone(); }}Copy the code
Client.java
public class Client {
// Prototype objects
private ConcretePrototype prototype = new ConcretePrototype();
// Clone objects
private ConcretePrototype clone;
public void operation(a) {
try {
clone = (ConcretePrototype) prototype.clone();
} catch (CloneNotSupportedException e) {
e.printStackTrace();
}
// false
System.out.println(clone == prototype);
// true
System.out.println(clone.getClass() == prototype.getClass());
// false: equals() is not defined
System.out.println(clone.equals(prototype));
}
public static void main(String[] args) {
newClient().operation(); }}Copy the code
The advantages and disadvantages
You can clone objects without coupling them to the specific class they belong to.
➤ When creating new object instances is complex, the prototype pattern can simplify the object creation process and improve the efficiency of creating new instances from an existing instance.
You can save the state of your object by using deep cloning.
needs a clone method for each class. If you modify an existing class, you must modify its source code, violating the “open close principle.”
The implementation of deep cloning requires more complex code.
Applicable scenario
Prototyping is recommended when:
(1) Creating new objects costs a lot.
(2) If the system wants to save the state of the object, and the state change of the object is very small, or the object itself occupies little memory, you can also use the “prototype mode” with the “memo mode” to apply. Conversely, if the state of the object changes a lot, or the memory footprint of the object is large, it is better to use “state mode” than “prototype mode.”
“Prototype mode” landed
(1) Many software offers Ctrl + C copy and Ctrl + V paste operations that are practical applications of prototype mode (CV engineers exclaim).
(2) In Spring, users can also use the prototype mode to create new bean instances, so that every time you get a new instance generated by cloning, you can modify it without any impact on the original instance object.
Model extension
Depth of cloning
Typically, a class contains a number of member objects. When objects are cloned using archetypal mode, they can be divided into two forms: deep and shallow clones, depending on whether the member objects “reference objects” are also cloned.
There are three ways to “copy” objects in Java: object assignment (=), shallow clone (implements Cloneable), and deep clone (implements Serializable).
direct assignment =, shallow clone Cloneable, deep clone Serializable:
Direct assignment
Shallow clone
A deep clone
Shallow clone
Shallow clones need to implement Cloneable interface, and with the help of clone() method, the above implementation is shallow clones.
A deep clone
A deep clone implements the Serializable interface with the ByteArrayOutputStream and ByteArrayInputStream classes: An object can be cloned by reading it into a Stream (serialization) and then reading it out (deserialization).
Note that the object written to the stream is a copy of the original object, which still exists in the JVM!
Email.java
public class Email implements Serializable {
private Attachment attachment;
public Email(a) {
// TODO
}
public Object deepClone(a) throws IOException, ClassNotFoundException {
/ / the serialization
ByteArrayOutputStream baos = new ByteArrayOutputStream();
ObjectOutputStream oos = new ObjectOutputStream(baos);
oos.writeObject(this);
// deserialize
ByteArrayInputStream bais = new ByteArrayInputStream(baos.toByteArray());
ObjectInputStream ois = new ObjectInputStream(bais);
return ois.readObject();
}
public void display(a) {
// TODO
}
public Attachment getAttachment(a) {
returnattachment; }}Copy the code
Attachment.java
Member objects also need to implement serialization interfaces
public class Attachment implements Serializable {
public void download(a) {
// TODO}}Copy the code
Stereotype schema with stereotype registry
Create a centralized stereotype registry based on the stereotype pattern to store commonly used stereotypes.
You can create a new factory class to implement the registry, or add a static method to get the prototype in the prototype base class.
PrototypeManager key code:
ConcretePrototypeA1, ConcretePrototypeA2... ConcretePrototypeA1, ConcretePrototypeA2... It is convenient to use stereotype patterns with stereotype manager
public class PrototypeManager {
// prototype-pattern with prototypeManager
static Map<String, ConcretePrototypeA> prototypeManager;
// all types
static {
prototypeManager = new HashTable<>();
// ConcretePrototypeA1 & ConcretePrototypeA2 are ConcretePrototypeA's SubClass
prototypeManager.put("A1".new ConcretePrototypeA1());
prototypeManager.put("A2".new ConcretePrototypeA2());
}
public static void add(String key, ConcretePrototypeA prototype) {
customerPrototypeManager.put(key, prototype);
}
public static ConcretePrototypeA get(String key) {
try {
return (ConcretePrototypeA) prototypeManager.get(key).clone();
} catch (CloneNotSupportedException e) {
e.printStackTrace();
}
return null; }}Copy the code
The last
Previous article: “Design Patterns” Builder
Next article: Continuing article, please look forward to…
Good code without explanation, pay attention to the “rip design patterns” column, with me to learn design patterns, your code can be as elegant as poetry!
/ END/If this article is helpful to you, click a “like” to support it, your support is my biggest motivation!