Java设计模式菜鸟系列(十六)原型模式建模与实现

转载请注明出处:http://blog.csdn.net/lhy_ycu/article/details/39997337

原型模式(Prototype):该模式的思想就是将一个对象作为原型,对其进行复制、克隆,产生一个和原对象类似的新对象。而这里的复制有两种:浅复制、深复制。

浅复制:将一个对象复制后,基本数据类型的变量都会重新创建,而引用类型,指向的还是原对象所指向的。

深复制:将一个对象复制后,不论是基本数据类型还有引用类型,都是重新创建的。简单来说,就是深复制进行了完全彻底的复制,而浅复制不彻底。

下面通过示例进行说明:

一、浅复制

1、uml建模:

2、代码实现

/**
 * 原型模式:将一个对象作为原型,对其进行复制、克隆,产生一个和原对象类似的【新对象】。
 *
 * 而这里的复制有两种:浅复制、深复制
 *
 * 示例(一) 浅复制:将一个对象复制后,基本数据类型的变量都会重新创建,
 *
 * 而引用类型,指向的还是原对象所指向的,【不会重新创建】。
 */
class Prototype implements Cloneable {
	private int age;
	private int[] array = new int[] { 1, 2, 3 };

	public Prototype() {
	}

	public Prototype(int age) {
		this.age = age;
	}

	public int getAge() {
		return age;
	}

	public void setAge(int age) {
		this.age = age;
	}

	public int[] getArray() {
		return array;
	}

	public void setArray(int[] array) {
		this.array = array;
	}

	/**
	 * 因为Cloneable接口是个空接口
	 *
	 * 此处的重点是super.clone()这句话,super.clone()调用的是Object的clone()方法,而在Object类中,clone()是native的
	 *
	 * 这就涉及到JNI,关于JNI还有NDK以后会讲到,这里你只要记住浅复制的核心是super.clone()。
	 */
	public Object cloneObject() throws CloneNotSupportedException {
		Prototype prototype = (Prototype) super.clone();
		return prototype;
	}
}

/**
 * 客户端测试类
 *
 * @author Leo
 */
public class Test {
	public static void main(String[] args) throws CloneNotSupportedException {
		Prototype prototype = new Prototype(20);
		Prototype cloneProto = (Prototype) prototype.cloneObject();
		/**
		 * 通过打印可以看到:prototype和cloneProto这两个同一类型的变量指向的是两个不同的内存地址
		 *
		 * 这说明克隆成功
		 */
		System.out.println("prototype = " + prototype);
		System.out.println("cloneProto = " + cloneProto);
		/**
		 * 要完全复制一个对象的话,那么它的引用类型变量array指向的肯定是不同的内存地址
		 *
		 * 而这里的引用类型变量array,指向的还是原对象所指向的。可以看到打印的内存地址是相同的。
		 *
		 * 这说明对象复制不彻底
		 */
		System.out.println("prototype.getArray() = " + prototype.getArray());
		System.out.println("cloneProto.getArray() = " + cloneProto.getArray());
		/**
		 * 透过这个例子可以看到:浅复制并没有将对象进行完全复制
		 */
	}
}

二、深复制

1、uml建模:

2、代码实现

/**
 * 示例(二) 深复制:将一个对象复制后,不论是基本数据类型还有引用类型,都是【重新创建】的。
 *
 * 简单来说,就是深复制进行了完全彻底的复制,而浅复制不彻底。
 *
 * 由于这里涉及到对对象的读写,所以这里用到了对象的序列化--实现了Serializable接口
 */
class Prototype implements Cloneable, Serializable {
	private static final long serialVersionUID = 1L;
	private int age;
	private int[] array = new int[] { 1, 2, 3 };

	public Prototype() {
	}

	public Prototype(int age) {
		this.age = age;
	}

	public int getAge() {
		return age;
	}

	public void setAge(int age) {
		this.age = age;
	}

	public int[] getArray() {
		return array;
	}

	public void setArray(int[] array) {
		this.array = array;
	}

	/* 深复制 */
	public Object deepClone() throws IOException, ClassNotFoundException {

		/* 写入当前对象的二进制流 */
		ByteArrayOutputStream bos = new ByteArrayOutputStream();
		ObjectOutputStream oos = new ObjectOutputStream(bos);
		oos.writeObject(this);

		/* 读出二进制流产生的新对象 */
		ByteArrayInputStream bis = new ByteArrayInputStream(bos.toByteArray());
		ObjectInputStream ois = new ObjectInputStream(bis);
		return ois.readObject();
	}

}

/**
 * 客户端测试类
 *
 * @author Leo
 */
public class Test {
	public static void main(String[] args) throws IOException,
			ClassNotFoundException {
		Prototype prototype = new Prototype(20);
		Prototype cloneProto = (Prototype) prototype.deepClone();
		/**
		 * 通过打印可以看到:prototype和cloneProto这两个同一类型的变量指向的是两个不同的内存地址
		 *
		 * 这说明克隆成功
		 */
		System.out.println("prototype = " + prototype);
		System.out.println("cloneProto = " + cloneProto);
		/**
		 * 通过打印可以看到,两个对象的引用类型变量array指向的是不同的内存地址
		 *
		 * 这说明对象进行了完全彻底的复制
		 */
		System.out.println("prototype.getArray() = " + prototype.getArray());
		System.out.println("cloneProto.getArray() = " + cloneProto.getArray());

		/**
		 * 当然我们也可以试着打印一下引用变量的内容,
		 *
		 * 可以看到:内容是不变的(1 2 3),改变的只是引用变量指向的内存地址。
		 */
		int[] proArray = prototype.getArray();
		int[] cloneProtoArray = cloneProto.getArray();
		for (int p : proArray) {
			System.out.print(p + "\t");
		}
		System.out.println();
		for (int p : cloneProtoArray) {
			System.out.print(p + "\t");
		}

	}
}

三、总结

1、浅复制的核心是super.clone(),它调用的是Object的clone()方法,而在Object类中,clone()是native的。

2、要实现深复制,需要采用二进制流的形式写入当前对象,再对其进行读取。

时间: 2024-10-25 17:57:44

Java设计模式菜鸟系列(十六)原型模式建模与实现的相关文章

Java设计模式菜鸟系列(十九)备忘录模式建模与实现

转载请注明出处:http://blog.csdn.net/lhy_ycu/article/details/40018967 备忘录模式(Memento): 主要目的是保存一个对象的某个状态,以便在适当的时候恢复对象. 一.uml建模: 二.代码实现 /** * 备忘录模式(Memento):主要目的是保存一个对象的某个状态,以便在适当的时候恢复对象 * * 示例:原始类--> 创建.恢复备忘录 */ class Original { private String state; public Or

Java设计模式菜鸟系列(十五)建造者模式建模与实现

转载请注明出处:http://blog.csdn.net/lhy_ycu/article/details/39856435 建造者模式(Builder):工厂类模式提供的是创建单个类的模式,而建造者模式则是将各种产品集中起来进行管理.简单起见,就拿之前的工厂方法模式进行修改一下就可以得到建造者模式. 一.uml建模: 二.代码实现 /** * 示例:建造者模式 * * 与工厂模式的区别:工厂类模式提供的是创建单个类,而建造者模式则是将各种产品集中起来进行管理 */ interface Sende

Java设计模式菜鸟系列(十四)代理模式建模与实现

转载请注明出处:http://blog.csdn.net/lhy_ycu/article/details/39856435 代理模式(Composite):代理模式其实就是多一个代理类出来,替原对象进行一些操作.比如咱有的时候打官司需要请律师,因为律师在法律方面有专长,可以替咱进行操作表达咱的想法,这就是代理的意思.代理模式分为两类:1.静态代理(不使用jdk里面的方法):2.动态代理(使用jdk里面的InvocationHandler和Proxy).下面请看示例: 一.静态代理 1.uml建模

Java设计模式菜鸟系列(十二)组合模式建模与实现

转载请注明出处:http://blog.csdn.net/lhy_ycu/article/details/39828653 组合模式(Composite):组合模式有时又叫部分-整体模式,将对象组合成树形结构来表示"部分-整体"层次结构.组合模式在处理树形结构的问题时比较方便. 一.uml建模: 二.代码实现 /** * 示例:组合模式有时也称"整合-部分"模式 * * 组合模式在处理树形结构的问题时比较方便 * * 节点 */ class TreeNode { /

Java设计模式菜鸟系列(三)装饰者模式建模与实现

转载请注明出处:http://blog.csdn.net/lhy_ycu/article/details/39759199 装饰者(Decorator)模式:动态地将责任附加到对象上,若要扩展功能,装饰者提供了比继承更具有弹性的替代方案.对于装饰者模式,它其实是一种包装,所以我更愿意称它为一种包装.像咱们以前经常使用的Java里面的IO流就用到了装饰者模式.比如:BufferedReader br = new BufferedReader(new InputStreamReader(new Fi

Java设计模式菜鸟系列(四)工厂方法模式建模与实现

转载请注明出处:http://blog.csdn.net/lhy_ycu/article/details/39760895 工厂方法模式(Factory Method) 工厂方法:顾名思义,就是调用工厂里的方法来生产对象(产品)的. 工厂方法实现方式有3种: 一.普通工厂模式.就是建立一个工厂类,对实现了同一接口的一些类进行实例的创建. 1.uml建模图: 2.代码实现 /** * 示例(一):普通工厂方法 * * 缺点:如果传递的字符串出错,则不能正确创建对象 */ interface Sen

Java设计模式菜鸟系列(五)抽象工厂模式建模与实现

转载请注明出处:http://blog.csdn.net/lhy_ycu/article/details/39784403 抽象工厂模式(Abstract Factory):抽象工厂--顾名思义,就是把工厂抽象出来,不同的工厂生产不同的产品.这样做有个好处:一旦需要增加新的功能,直接增加新的工厂类就可以了,不需要修改之前的代码. 一.uml建模图: 二.代码实现 /** * 示例:抽象工厂--顾名思义,就是把工厂抽象出来,不同的工厂生产不同的产品 * * 优点:一旦需要增加新的功能,直接增加新的

Java设计模式菜鸟系列(二十三)访问者模式建模与实现

转载请注明出处:http://blog.csdn.net/lhy_ycu/article/details/40028509 访问者模式(Visitor):把数据结构和作用于结构上的操作解耦合,使得操作集合可相对自由地演化.访问者模式适用于数据结构相对稳定而算法又容易变化的系统.访问者模式的优点是增加操作很容易,因为增加操作意味着增加新的访问者:而它的缺点就是增加新的数据结构很困难. 一.uml建模: 二.代码实现 /** * 访问者模式(Visitor):把数据结构和作用于结构上的操作解耦合,使

Java设计模式菜鸟系列(十)模板方法模式建模与实现

转载请注明出处:http://blog.csdn.net/lhy_ycu/article/details/39806973 模板方法模式(Template Method):在一个方法中定义了一个算法的骨架,而将一些步骤延迟到子类中.模板方法使得子类可以再不改变算法结构的情况下,重新定义算法中的某些步骤.简而言之:模板方法定义了一个算法的步骤,并允许子类为一个或多个步骤提供实现. 一.uml建模: 二.代码实现: /** * 示例:模板方法定义了一个算法的步骤,并允许子类为一个或多个步骤提供实现.