• java设计模式之原型模式


    一:原型模式

    1.什么是原型模式?
    用一个已经创建的实例作为原型,通过复制该原型对象来创建一个和原型对象相同的新对象。
    基本介绍
    1.原型模式(Prototype模式)是指:用原型实例指定创建对象的种类,并且通过拷贝这些原型,创建新的对象
    2.原型模式是一种创建型设计模式,允许一个对象再创建另外一个可定制的对象,无需知道如何创建的细节
    3.工作原理是:通过将一个原型对象传给那个要发动创建的对象,这个要发动创建的对象通过请求原型对象拷贝它们自己来实施创建,即 对象.clone()
    4.形象的理解:孙大圣拔出猴毛, 变出其它孙大圣

    2.原型模式适合应用场景
    1.如果你需要复制一些对象, 同时又希望代码独立于这些对象所属的具体类, 可以使用原型模式。
    2.如果子类的区别仅在于其对象的初始化方式, 那么你可以使用该模式来减少子类的数量。 别人创建这些子类的目的可能是为了创建特定类型的对象。
    4.Spring中原型bean的创建,就是原型模式的应用

    3. 原型模式优缺点
    优点:
    1.你可以克隆对象, 而无需与它们所属的具体类相耦合。
    2.你可以克隆预生成原型, 避免反复运行初始化代码。
    3.你可以更方便地生成复杂对象。
    4.你可以用继承以外的方式来处理复杂对象的不同配置。
    缺点:
    1.克隆包含循环引用的复杂对象可能会非常麻烦。

    4.原型模式实现
    原型模式的克隆分为浅克隆和深克隆。
    浅克隆:创建一个新对象,新对象的属性和原来对象完全相同,对于非基本类型属性,仍指向原
    有属性所指向的对象的内存地址。
    深克隆:创建一个新对象,属性中引用的其他对象也会被克隆,不再指向原有对象地址。

    二:原型模式代码实现

    原型模式包含如下角色:
    抽象原型类:规定了具体原型对象必须实现的的 clone() 方法。
    具体原型类:实现抽象原型类的 clone() 方法,它是可被复制的对象。
    访问类:使用具体原型类中的 clone() 方法来复制新的对象。
    浅克隆

    //具体原型类
    public class Sheep implements Cloneable {
    	private String name;
    	private int age;
    	private String color;
    	private String address = "蒙古羊";
    	public Sheep friend; //是对象, 克隆是会如何处理
    	public Sheep(String name, int age, String color) {
    		super();
    		this.name = name;
    		this.age = age;
    		this.color = color;
    	}
    	public String getName() {
    		return name;
    	}
    	public void setName(String name) {
    		this.name = name;
    	}
    	public int getAge() {
    		return age;
    	}
    	public void setAge(int age) {
    		this.age = age;
    	}
    	public String getColor() {
    		return color;
    	}
    	public void setColor(String color) {
    		this.color = color;
    	}
    	
    	
    	
    	@Override
    	public String toString() {
    		return "Sheep [name=" + name + ", age=" + age + ", color=" + color + ", address=" + address + "]";
    	}
    	//克隆该实例
    	//重写clone方法,或者用默认的clone方法也行,只不过调用时候需要抛异常
    	@Override
    	protected Object clone()  {
    		Sheep sheep = null;
    		try {
    			sheep = (Sheep)super.clone();
    		} catch (Exception e) {
    			System.out.println(e.getMessage());
    		}
    		return sheep;
    	}
    }
    
    
    //--访问类
    	//可以看出都是克隆的原型,调用原型类的clone来复制对象
    	public static void main(String[] args)  {
    		System.out.println("原型模式完成对象的创建");
    	
    		Sheep sheep = new Sheep("tom", 1, "白色");
    
    		sheep.friend = new Sheep("jack", 2, "黑色");
    
    		Sheep sheep2 = (Sheep)sheep.clone(); //克隆
    		Sheep sheep3 = (Sheep)sheep.clone(); //克隆
    		Sheep sheep4 = (Sheep)sheep.clone(); //克隆
    		Sheep sheep5 = (Sheep)sheep.clone(); //克隆
    		
    		System.out.println("sheep2 =" + sheep2 + "sheep2.friend=" + sheep2.friend.hashCode());
    		System.out.println("sheep3 =" + sheep3 + "sheep3.friend=" + sheep3.friend.hashCode());
    		System.out.println("sheep4 =" + sheep4 + "sheep4.friend=" + sheep4.friend.hashCode());
    		System.out.println("sheep5 =" + sheep5 + "sheep5.friend=" + sheep5.friend.hashCode());
    	}
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48
    • 49
    • 50
    • 51
    • 52
    • 53
    • 54
    • 55
    • 56
    • 57
    • 58
    • 59
    • 60
    • 61
    • 62
    • 63
    • 64
    • 65
    • 66
    • 67
    • 68
    • 69
    • 70
    • 71
    • 72

    在这里插入图片描述

    深克隆(原型引用类型,并且原型和引用类型都需要实现序列化,和实现克隆接口)

    //原型
    public class DeepProtoType implements Serializable, Cloneable{
    	
    	public String name; //String 属性
    	public DeepCloneableTarget deepCloneableTarget;// 引用类型
    	public DeepProtoType() {
    		super();
    	}
    	
    	
    	//深拷贝 - 方式 1 使用clone 方法
    	@Override
    	protected Object clone() throws CloneNotSupportedException {
    		
    		Object deep = null;
    		//这里完成对基本数据类型(属性)和String的克隆
    		deep = super.clone(); 
    		
    		//克隆引用类型,对引用类型的属性,进行单独处理
    		DeepProtoType deepProtoType = (DeepProtoType)deep;
    		deepProtoType.deepCloneableTarget  = (DeepCloneableTarget)deepCloneableTarget.clone();
    		
    		return deepProtoType;
    	}
    	
    	//深拷贝 - 方式2 通过对象的序列化实现 (推荐)
    	public Object deepClone() {
    		
    		//创建流对象
    		ByteArrayOutputStream bos = null;
    		ObjectOutputStream oos = null;
    		ByteArrayInputStream bis = null;
    		ObjectInputStream ois = null;
    		
    		try {
    			
    			//序列化
    			bos = new ByteArrayOutputStream();
    			oos = new ObjectOutputStream(bos);
    			oos.writeObject(this); //当前这个对象以对象流的方式输出
    			
    			//反序列化
    			bis = new ByteArrayInputStream(bos.toByteArray());
    			ois = new ObjectInputStream(bis);
    			DeepProtoType copyObj = (DeepProtoType)ois.readObject();
    			
    			return copyObj;
    			
    		} catch (Exception e) {
    			e.printStackTrace();
    			return null;
    		} finally {
    			//关闭流
    			try {
    				bos.close();
    				oos.close();
    				bis.close();
    				ois.close();
    			} catch (Exception e2) {
    				System.out.println(e2.getMessage());
    			}
    		}
    	}
    	
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48
    • 49
    • 50
    • 51
    • 52
    • 53
    • 54
    • 55
    • 56
    • 57
    • 58
    • 59
    • 60
    • 61
    • 62
    • 63
    • 64
    • 65
    //引用类型
    public class DeepCloneableTarget implements Serializable, Cloneable {
    	
    	/**
    	 * 
    	 */
    	private static final long serialVersionUID = 1L;
    
    	private String cloneName;
    
    	private String cloneClass;
    
    	//构造器
    	public DeepCloneableTarget(String cloneName, String cloneClass) {
    		this.cloneName = cloneName;
    		this.cloneClass = cloneClass;
    	}
    
    	//因为该类的属性,都是String , 因此我们这里使用默认的clone完成即可
    	@Override
    	protected Object clone() throws CloneNotSupportedException {
    		return super.clone();
    	}
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    	//访问类(深克隆的内存对象是不一致的)
    	public static void main(String[] args) throws Exception {
    		DeepProtoType p = new DeepProtoType();
    		p.name = "宋江";
    		p.deepCloneableTarget = new DeepCloneableTarget("大牛", "小牛");
    
    		
    		//方式1 完成深拷贝
    		
    //		DeepProtoType p2 = (DeepProtoType) p.clone();
    //		
    //		System.out.println("p.name=" + p.name + "p.deepCloneableTarget=" + p.deepCloneableTarget.hashCode());
    //		System.out.println("p2.name=" + p.name + "p2.deepCloneableTarget=" + p2.deepCloneableTarget.hashCode());
    	
    		//方式2 完成深拷贝
    		DeepProtoType p2 = (DeepProtoType) p.deepClone();
    		
    		System.out.println("p.name=" + p.name + "p.deepCloneableTarget=" + p.deepCloneableTarget.hashCode());
    		System.out.println("p2.name=" + p.name + "p2.deepCloneableTarget=" + p2.deepCloneableTarget.hashCode());
    	
    	}
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21

    在这里插入图片描述

  • 相关阅读:
    #django基本常识01#
    C++ 第一阶段编程练习
    拼多多根据关键词取商品列表 API 返回值说明
    「Python实用秘技07」在pandas中实现自然顺序排序
    私有化部署的知识管理平台对企业有什么意义?
    数据结构之链表详解(2)——双向链表
    搭建双主双从的MySQL主从同步复制
    企业工程项目管理系统源码(三控:进度组织、质量安全、预算资金成本、二平台:招采、设计管理)
    netty面试题及答案
    Java急速转职GoLang工程师资料
  • 原文地址:https://blog.csdn.net/m0_47944994/article/details/127884565