1.概述
原型模式是用于创建重复的对象,同时又能保证性能,它提供了一种创建对象的最佳方式。当直接创建对象的代价比较大时,则采用这种模式。
使用原型模式复制对象不会调用类的构造方法。所以,单例模式原型模式是冲突的,在使用时需要特别注意。在原型模式中,成员变量不能被final修饰。
2.关键步骤
a.实现一个原型接口。
b.重写该接口的clone方法
3.优缺点
优点:
使用原型模式创建对象比直接new一个对象更有效。
隐藏制造新实例的复杂性。
重复的创建相似的对象可以考虑使用原型模式。
缺点:
每一个类必须配备一个克隆方法
深层复制比较复杂
4.浅拷贝与深拷贝
浅拷贝:
仅仅只是对内存地址进行复制。Object类的clone方法只会拷贝对象中的基本数据类型,对于数组,容器对象,引用对象都不会拷贝,这就是浅拷贝。
深拷贝:
重新申请一个新的内存地址。若要实现深拷贝,必须将原型模式中的数组,容器对象,引用对象等进行另行拷贝。
应用实例:通过原型设计模式来模拟发送广告信的效果
案例1—未引入原型模式
AdvTemplate.java
package demo_prototype01;
//广告信
public class AdvTemplate {
//广告信名称
private String advName="广告信名称";
//广告信内容
private String advContent="广告信内容";
public String getAdvName() {
return advName;
}
public void setAdvName(String advName) {
this.advName = advName;
}
public String getAdvContent() {
return advContent;
}
public void setAdvContent(String advContent) {
this.advContent = advContent;
}
}
Mail.java
package demo_prototype01;
public class Mail {
//收件人
private String recevier;
//邮件名称
private String name;
//邮件内容
private String content;
//单参构造方法
public Mail(AdvTemplate adv) {
this.content=adv.getAdvContent();
this.name=adv.getAdvName();
}
public String getRecevier() {
return recevier;
}
public void setRecevier(String recevier) {
this.recevier = recevier;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public String getContent() {
return content;
}
public void setContent(String content) {
this.content = content;
}
}
Test.java
package demo_prototype01;
import java.util.Random;
public class Test {
//设置邮件发送数
private static int MAX_COUNT=6;
public static void main(String[] args) {
int i=0;
//邮件创建
Mail mail=new Mail(new AdvTemplate());
while(i<MAX_COUNT) {
//设置收件人信箱
mail.setRecevier(getRandString(5)+"@"+getRandString(3)+".com");
//设置邮件内容
mail.setContent(getRandString(20));
//发送邮件
sendMail(mail);
i++;
}
}
//发送邮件
public static void sendMail(Mail mail) {
//打印邮件相关信息
System.out.println("标题:"+mail.getName()+"\t收件人:"
+mail.getRecevier()+"\t内容:"+mail.getContent()
+"\t...发送成功!");
}
//模拟生成字符串
public static String getRandString(int maxLength) {
String source="abcdefghijklmnopqrstuvwxyz";
//定义字符串缓冲区
StringBuilder sb=new StringBuilder();
//定义随机数组生成器
Random random=new Random();
//循环
for(int i=0;i<maxLength;i++) {
sb.append(source.charAt(random.nextInt(source.length())));
}
return sb.toString();
}
}
运行结果图
案例2—引入原型模式
AdvTemplate.java
package demo_prototype01;
//广告信
public class AdvTemplate {
//广告信名称
private String advName="广告信名称";
//广告信内容
private String advContent="广告信内容";
public String getAdvName() {
return advName;
}
public void setAdvName(String advName) {
this.advName = advName;
}
public String getAdvContent() {
return advContent;
}
public void setAdvContent(String advContent) {
this.advContent = advContent;
}
}
Mail.java
package demo_prototype02;
//1.实现接口
public class Mail implements Cloneable{
//收件人
private String recevier;
//邮件名称
private String name;
//邮件内容
private String content;
//单参构造方法
public Mail(AdvTemplate adv) {
this.content=adv.getAdvContent();
this.name=adv.getAdvName();
}
//2.重写clone方法
@Override
protected Object clone() {
Mail mail=null;
try {
mail=(Mail)super.clone();
} catch (CloneNotSupportedException e) {
e.printStackTrace();
}
return mail;
}
public String getRecevier() {
return recevier;
}
public void setRecevier(String recevier) {
this.recevier = recevier;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public String getContent() {
return content;
}
public void setContent(String content) {
this.content = content;
}
}
Test.java
package demo_prototype02;
import java.util.Random;
public class Test {
//设置邮件发送数
private static int MAX_COUNT=6;
public static void main(String[] args) {
int i=0;
//邮件创建
Mail mail=new Mail(new AdvTemplate());
while(i<MAX_COUNT) {
//克隆
Mail mailClone=(Mail)mail.clone();
//设置收件人信箱
mail.setRecevier(getRandString(5)+"@"+getRandString(3)+".com");
//设置邮件内容
mail.setContent(getRandString(20));
//发送邮件
sendMail(mailClone);
i++;
}
}
//发送邮件
public static void sendMail(Mail mail) {
//打印邮件相关信息
System.out.println("标题:"+mail.getName()+"\t收件人:"
+mail.getRecevier()+"\t内容:"+mail.getContent()
+"\t...发送成功!");
}
//模拟生成字符串
public static String getRandString(int maxLength) {
String source="abcdefghijklmnopqrstuvwxyz";
//定义字符串缓冲区
StringBuilder sb=new StringBuilder();
//定义随机数组生成器
Random random=new Random();
//循环
for(int i=0;i<maxLength;i++) {
sb.append(source.charAt(random.nextInt(source.length())));
}
return sb.toString();
}
}
运行结果图