在说原型模式之前,我们先来看java里面的深复制和浅复制:
1. 浅复制:被复制的对象的所有变量都持有和原来对象的变量相同的值,而所有的对其他对象的引用都指向原来的对象。
2.
深复制:被复制对象的所有变量都含有与原来对象相同的值,除去那些引用其他变量的对象。那些引用其他对象的变量将指向被复制过来的新对象,而不是原来那些被引用的对象。深复制需要把要复制的对象的所有引用都复制一遍。
这两者的区别就是关于引用对象的处理,浅复制是不考虑引用对象,而深复制需要考虑引用对象的问题。
对java中的clone()方法我们有以下约定:
1. 对于任何的对象x, 都有x.clone()!=x
2. 对于任何的对象x, 都有x.clone().getClass() == x.getClass();
3. 对于任何的对象x, 都有x.clone().equals(x) = true
继承自java.lang.Object的clone方法是浅复制。
在深复制的过程中我们需要考虑深复制的深度问题,可能会出现循环引用的问题,所以对于深复制,我们一般是利用串行化来做深复制的:先把对象写到流里,再从流里读出来。
下面是一段示意性的代码:
package com.javadesignpattern.prototype;import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.io.Serializable;public class TestDeepClone implements Serializable{
/**
*
*/
private static final long serialVersionUID = 1L;public static void main(String[] args) throws IOException, ClassNotFoundException{
File file = new File("out.ser");
FileOutputStream fos = new FileOutputStream(file);
ObjectOutputStream oos = new ObjectOutputStream(fos);
oos.writeObject("test");
oos.flush();FileInputStream fis = new FileInputStream(file);
ObjectInputStream ois = new ObjectInputStream(fis);
System.out.println(ois.readObject());
ois.close();
}}
下面看原型模式的结构,《java与模式》中给出两种原型模式,一种是简单形式的原型模式,一种是登记型的原型模式。
1. 简单形式的原型模式
示意性代码:
?
1 2 3 4 5 6 7 |
|
package com.javadesignpattern.prototype.simple;public class ConcretePrototype implements Prototype {
public Object clone(){
try {
return super.clone();
} catch (CloneNotSupportedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
return null;
}
}}
package com.javadesignpattern.prototype.simple;public class Client {
private Prototype prototype;
public void operation(Prototype example){
Prototype prototype = (Prototype)example.clone();
}public static void main(String[] args){
Prototype prototypeTest = new ConcretePrototype();
Client clientInctance = new Client();
clientInctance.operation(prototypeTest);
System.out.println(clientInctance.prototype);
}}
2. 登记形式的原型模式
示意性代码如下:
1. Prototype和CPrototype类和上面的简单原型模式没什么区别。
package com.javadesignpattern.prototype.register;import java.util.ArrayList;
public class PrototypeManager {
private ArrayList object = new ArrayList();
public Prototype get(int i) {
return (Prototype)object.get(i);
}public void set(Prototype objectp) {
object.add(objectp);
}public int getSize(){
return object.size();
}}
package com.javadesignpattern.prototype.register;public class Client {
private PrototypeManager mgr;
private Prototype prototype;public String registerPrototype(){
prototype = new ConcretePrototype();
mgr = new PrototypeManager();
mgr.set((Prototype)prototype.clone());
if(mgr != null ){
return "SUCCESS";
}else{
return "FAILED";
}
}public static void main(String[] args){
Client client = new Client();
System.out.println(client.registerPrototype() + "-------" + client.mgr.getSize());
}}
简单原型模式和登记类型的原型模式的比较:
1. 如果需要创建的原型对象的数目比较少并且比较固定的话,我们可以选择第一种简单类型的原型,这种情况下,原型对象由客户端持有。
2.
如果需要创建的原型对象不固定的话,可以选择第二种方式:登记类型的原型,这样的话原型对象交给manager类持有,客户端从原型类中解脱出来,在创建一个新的原型对象之前,客户端可以看Manager类中是不是已经有符合条件的对象,有的话,就直接从manager类中拿出来用,没有的话,客户端就自己再去复制一份。
description 数组的中文打印