【秒懂设计模式】原型模式

简介:  秒懂设计模式——原型模式 (五)原型模式 【官方定义】用原型实例指定创建对象的种类,并且通过拷贝这些原型,创建新的对象,属于对象创建模式。 【举例】在找工作的时候,通常需要我们的毕业证和学位证,但是我们一般不会把原件(原型实例)给到用人单位,而是会把原件复印(拷贝)几份,然后供用人单位使用。 【Java代码】 ①创建一个证书抽象类。 package com.liyan.

 秒懂设计模式——原型模式


(五)原型模式

【官方定义】用原型实例指定创建对象的种类,并且通过拷贝这些原型,创建新的对象,属于对象创建模式。

【举例】在找工作的时候,通常需要我们的毕业证和学位证,但是我们一般不会把原件(原型实例)给到用人单位,而是会把原件复印(拷贝)几份,然后供用人单位使用。

Java代码】

①创建一个证书抽象类。

package com.liyan.prototype;
/**
 * 证书抽象类
 * <p>Title: Certificate</p>  
 * @author  Liyan  
 * @date    2017年5月2日 下午3:17:37
 */
public abstract class Certificate implements Cloneable{
	
	/**证书编号*/
	private Integer id;
	
	/**证书名称*/
	protected String name;
	
	/**测试方法*/
	abstract void testMethod();
	
	public Integer getId() {
		return id;
	}

	public void setId(Integer id) {
		this.id = id;
	}

	public String getName() {
		return name;
	}

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

	@Override
	protected Object clone() throws CloneNotSupportedException {
		return super.clone();
	}

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

②创建证书扩展类——毕业证

package com.liyan.prototype;
/**
 * 数据信息扩展类——毕业证
 * <p>Title: GraduationCertificate</p>  
 * @author  Liyan  
 * @date    2017年5月2日 下午3:17:47
 */
public class GraduationCertificate extends Certificate{
	
	public GraduationCertificate() {
		name = "毕业证";
	}

	@Override
	void testMethod() {
		System.out.println("毕业证复印完成!");
	}

}

③创建证书扩展类——学位证

package com.liyan.prototype;
/**
 * 证书扩展类——学位证
 * <p>Title: DegreeCertificate</p>  
 * @author  Liyan  
 * @date    2017年5月2日 下午3:18:26
 */
public class DegreeCertificate extends Certificate{
	
	public DegreeCertificate() {
		name = "学位证";
	}
	
	@Override
	void testMethod() {
		System.out.println("学位证复印完成!");
	}

}

④创建原型模式管理器——证书管理器

package com.liyan.prototype;
import java.util.HashMap;

/**
 * 原型模式管理器——证书管理器
 * <p>Title: CertificateManger</p>  
 * @author  Liyan  
 * @date    2017年5月2日 下午3:24:40
 */
public class CertificateManger {
	//证书的map集合
	private static HashMap<Integer, Certificate> certificateMap = new HashMap<Integer, Certificate>(0);
	//私有构造方法
	private CertificateManger (){}
	//懒加载构建原型模式管理器的单例模式
	private static CertificateManger certificateManger = null;
	
	/**
	 * 获取单例模式的原型模式管理器
	 * <p>Title: getCertificateManger</p>  
	 * @author Liyan  
	 * @date   2017年5月2日 下午3:24:51  
	 * @return CertificateManger
	 */
	public static CertificateManger getCertificateManger() {
		if (certificateManger == null) {
			certificateManger = new CertificateManger();
		}
		return certificateManger;
	}
	
	/**
	 * 加载缓存
	 * <p>Title: getLoad</p>  
	 * @author Liyan  
	 * @date   2017年5月2日 下午3:25:07        
	 */
	public static void getLoad() {
		GraduationCertificate certificate1 = new GraduationCertificate();
		certificateMap.put(1, certificate1);
		DegreeCertificate certificate2 = new DegreeCertificate();
		certificateMap.put(2, certificate2);
	}  
	
	/**
	 * 根据key获取克隆后的对象
	 * <p>Title: getCertificate</p>  
	 * @author Liyan  
	 * @date   2017年5月2日 下午3:25:31  
	 * @param key
	 * @return Certificate
	 */
	public static Certificate getCertificate(Integer key) {
		try {
			Certificate certificate = (Certificate) certificateMap.get(key);
			if (certificate != null) {
				Certificate datinfo = (Certificate) certificate.clone();
				System.out.println("key="+key+"的对象克隆完成,datinfo="+datinfo);
				return datinfo;
			}else {
				System.out.println("key="+key+"并未加载!");
				return null;
			}
		} catch (CloneNotSupportedException e) {
			e.printStackTrace();
		}
		return null;
	}

}

⑤测试

package com.liyan.prototype;
/**
 * 测试
 * <p>Title: Test</p>  
 * @author  Liyan  
 * @date    2017年5月2日 下午3:30:21
 */
public class Test {
	@SuppressWarnings("static-access")
	public static void main(String[] args) {
		//提前加载到缓存数据
		CertificateManger.getLoad();
		//获取单例类CertificateManger
		CertificateManger manger  = CertificateManger.getCertificateManger();
		//获取毕业证复印件
		Certificate certificate1 = manger.getCertificate(1);
		certificate1.testMethod();
		//获取学位证复印件
		Certificate certificate2 = manger.getCertificate(2);
		certificate2.testMethod();
	}
}

⑥测试结果

key=1的对象克隆完成,datinfo=Certificate [id=null, name=毕业证]
毕业证复印完成!
key=2的对象克隆完成,datinfo=Certificate [id=null, name=学位证]
学位证复印完成!

图解关系:

 

分析:

这种模式是实现了一个原型接口,该接口用于创建当前对象的克隆。当直接创建对象的代价比较大时,则采用这种模式。

独特的优点

1)性能比直接new一个对象要好的多。因为clone方法是一个本地方法,它直接操作内存中的二进制流,特别是复制大对象时,性能的差别非常明显,所以在需要重复地创建相似对象时可以考虑使用原型模式。

2)原型模式创建模式不会损坏原有对象

3)工厂模式对新产品的适应能力比较弱,层级结构比较复杂,没有原型模式简洁

注意事项:

1)使用原型模式复制对象不会调用类的构造方法。因为clone方法,会直接在内存中复制数据。

2)单例模式与原型模式是冲突的。因为clone方法直接无视构造方法的权限。





相关文章
|
7月前
|
设计模式 安全 Java
面向对象编程的精髓:Java设计模式 - 原型模式(Prototype)完全参考手册
【4月更文挑战第7天】原型模式是OOP中的创建型设计模式,用于通过复制现有实例创建新实例,尤其适用于创建成本高或依赖其他对象的情况。它包括Prototype接口、ConcretePrototype实现和Client客户端角色。优点是性能优化、避免子类化和动态增加产品族。实现包括定义原型接口、实现具体原型和客户端调用克隆方法。最佳实践涉及确保克隆正确性、选择深拷贝或浅拷贝及考虑线程安全。但需注意克隆方法管理、性能开销和循环引用等问题。在Java中,实现Cloneable接口和覆盖clone方法可实现原型模式。
86 4
|
7月前
|
设计模式 Java 关系型数据库
23种设计模式 —— 原型模式【克隆羊、浅拷贝、深拷贝】
23种设计模式 —— 原型模式【克隆羊、浅拷贝、深拷贝】
|
7月前
|
设计模式 安全 Java
【设计模式】原型模式
【设计模式】原型模式
|
3月前
|
设计模式 Java 关系型数据库
【Java笔记+踩坑】设计模式——原型模式
对比原型模式和传统方式的实现思路、代码方案、优缺点,阐述原型模式的使用场景,以及深拷贝、浅拷贝等相关概念,并扩展原型模式在Spring源码中的应用。
【Java笔记+踩坑】设计模式——原型模式
|
3月前
|
设计模式 Java
Java设计模式-原型模式(3)
Java设计模式-原型模式(3)
Java设计模式-原型模式(3)
|
5月前
|
设计模式
iLogtail设计模式问题之iLogtail中的原型模式是什么
iLogtail设计模式问题之iLogtail中的原型模式是什么
iLogtail设计模式问题之iLogtail中的原型模式是什么
|
5月前
|
设计模式 JavaScript
js设计模式【详解】—— 原型模式
js设计模式【详解】—— 原型模式
53 6
|
6月前
|
设计模式 Java
Java设计模式之原型模式详解
Java设计模式之原型模式详解
|
6月前
|
设计模式
原型模式-大话设计模式
原型模式-大话设计模式
|
6月前
|
设计模式 Java Spring
设计模式——原型模式
设计模式——原型模式