原型模式_黑马程序员学习笔记

1.概述

用一个已经创建的实例为原型,通过复制该原型对象创建一个和原型对象相同的新对象

2.结构 

原型模式包含如下角色:

  • 抽象原型类:规定了具体原型对象必须实现的的 clone() 方法
  • 具体原型类:实现抽象原型类的 clone() 方法,它是可被复制的对象。
  • 访问类:使用具体原型类中的 clone() 方法来复制新的对象。

 

 3.实现

原型模式的克隆分为浅克隆和深克隆。

浅克隆:创建一个新对象,新对象的属性和原来对象完全相同,对于非基本类型属性,仍指向原有属性所指向的对象的内存地址。

深克隆:创建一个新对象,属性中引用的其他对象也会被克隆,不再指向原有对象地址

Java中的Object类中提供了 clone() 方法来实现浅克隆。 Cloneable 接口是上面的类图中的抽象原型类,而实现了Cloneable接口的子实现类就是具体的原型类。代码如下:

Realizetype(具体的原型类):

public class RealizeType implements Cloneable{

	
	public RealizeType() {
		// TODO Auto-generated constructor stub
		System.out.println("具体原型对象创建成功");
	}
	public RealizeType clone() {
		System.out.println("具体原型复制成功");
		try {
			return (RealizeType) super.clone();
		} catch (CloneNotSupportedException e) {
			
			e.printStackTrace();
		}
		return null;
		
	}
}

PrototypeTest(测试访问类):

 

public class Client {

	public static void main(String[] args) {
//		1.创建原型类对象
		RealizeType realizeType=new RealizeType();
//      2.调用Realizetype类中的clone方法进行克隆
		RealizeType clone=realizeType.clone();
		System.out.println("原型类对象的地址:"+realizeType);
		System.out.println("克隆而来的对象的地址"+clone);
		System.out.println("两对象是否相等"+(realizeType==clone));
	}
}

运行结果:

 4.例子

 用原型模式生成“三好学生”奖状

同一学校的“三好学生”奖状除了获奖人姓名不同,其他都相同,可以使用原型模式复制多个“三好学生”奖状出来,然后在修改奖状上的名字即可。

类图如下:

 奖状类

/**
 * 奖状类
 * @author CharlieLiang
 *
 */
public class Citation implements Cloneable{

	private String name;

	public String getName() {
		return name;
	}

	public void setName(String name) {
		this.name = name;
	}
	
	public Citation Clone () {
		try {
			return (Citation) super.clone();
		} catch (CloneNotSupportedException e) {
			
			e.printStackTrace();
		}
		return null;
	}
	
	public void show() {
		System.out.println(name + "同学:在2024学年第一学期中表现优秀,被评为三好学生。特发此状!");
	}
}
public class CitationTest {

	public static void main(String[] args) {
		//1.创建原型对象
		Citation citation=new Citation();
		
		//2.克隆奖状对象
		Citation c1=citation.Clone();
		Citation c2=citation.Clone();
		
		c1.setName("张三");
		c2.setName("李四");
		
		//3.调用show显示
		c1.show();
		c2.show();
	}
}

运行结果:

 5. 使用场景

· 对象的创建非常复杂,可以使用原型模式快捷的创建对象

·性能和安全要求比较高

 6. 深克隆

 将上面的“三好学生”奖状的案例中Citation类的name属性修改为Student类型的属性。代码如下:

/**
 * 学生类
 * @author CharlieLiang
 *
 */
public class Student {

	private String name;

	public String getName() {
		return name;
	}

	public void setName(String name) {
		this.name = name;
	}

	@Override
	public String toString() {
		return "Student [name=" + name + "]";
	}
	
}

 

/**
 * 奖状类
 * @author CharlieLiang
 *
 */
public class Citation implements Cloneable{

	private Student stu;

	public Student getStu() {
		return stu;
	}

	public void setStu(Student stu) {
		this.stu = stu;
	}
	
	
	public Citation Clone() {
		try {
			return (Citation) super.clone();
		} catch (CloneNotSupportedException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
		return null;
	}
	
	public void show() {
		System.out.println(stu.getName()+"同学:在2024学年第一学期中表现优秀,被评为三好学生。特发此状!");
	}
}
/**
 * 测试类
 * @author CharlieLiang
 *
 */
public class CitationTest {

	public static void main(String[] args) {
		//1.创建原型对象
		Citation c1=new Citation();
		Student stu=new Student();
		stu.setName("张三");
		c1.setStu(stu);
		
		
		//2.克隆奖状对象
		Citation c2=c1.Clone();
		Student stu2=c2.getStu();
		stu2.setName("李四");
		
		//判断stu对象和stu1对象是否同一个对象
		System.out.println("stu和stu2是否同一个对象 : "+(stu==stu2));
		
		
		//3.调用show显示
		c1.show();
		c2.show();
	}
}

运行结果:

 

 说明:

stu对象和stu2对象是同一个对象,就会产生将stu对象中name属性值改为“李四”,两个Citation(奖状)对象中显示的都是李四。这就是浅克隆的效果,对具体原型类(Citation)中的引用类型的属性进行引用的复制。这种情况需要使用深克隆,而进行深克隆需要使用对象流。代码如下:

奖状类

import java.io.Serializable;

public class Citation implements Cloneable,Serializable{

	private Student stu;

	public Student getStu() {
		return stu;
	}

	public void setStu(Student stu) {
		this.stu = stu;
	}
	
	
	public Citation Clone() {
		try {
			return (Citation) super.clone();
		} catch (CloneNotSupportedException e) {
			
			e.printStackTrace();
		}
		return null;
	}
	
	public void show() {
		System.out.println(stu.getName()+"同学:在2024学年第一学期中表现优秀,被评为三好学生。特发此状!");
	}
}

 测试类

import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;

public class CitationTest {

	public static void main(String[] args) throws Exception {
		//1.创建原型对象
		Citation c1=new Citation();
		
		
		Student stu=new Student();
		stu.setName("张三");
		c1.setStu(stu);
		
		//创建对象输出流对象
		ObjectOutputStream oos=new ObjectOutputStream(
				new FileOutputStream("a.txt"));
		//写对象
		oos.writeObject(c1);
		oos.close();
		
		//创建对象输入流对象
		ObjectInputStream ois=new ObjectInputStream(
				new FileInputStream("a.txt"));
		
		Citation c2=(Citation) ois.readObject();
		Student stu2=c2.getStu();
		stu2.setName("李四");
		
		c1.show();
		c2.show();
	}
}

运行结果:

 

 注意:Citation类和Student类必须实现Serializable接口,否则会抛NotSerializableException异常。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值