**1. 定义:通过复制现有实例来创建新的实例,无需知道相应类的信息。**原理示例如下:
具体代码如下:
2. 原型模式优缺点:
**优点:**通过原型模式是从内存中直接拷贝一份,隐藏了创建新实例的复杂性;相对于new一个对象,速度更快,更高效。重复创建相似的对象的时候,可以考虑使用原型模式。
**缺点:**每一个类必须配备一个克隆方法。深层复制比较复杂。
3. 单例模式与原型模式的冲突
比如,在一个类A中,如果该类的构造函数是private,而且A继承了Cloneable这个类,那么,等于破坏了单例模式。因为原型模式的克隆,绕过了**单例模式的“外部不可通过类的private构造函数构造一个类”**这个原则,从而破坏了单例模式。所以,使用克隆的类,不可以继承Cloneable这个类。代码示例如下:
public class EventTemplate {
private String eventSubject, eventContent;
public EventTemplate(String eventSubject, String eventContent) {
this.eventSubject = eventSubject;
this.eventContent = eventContent;
}
public String geteventSubject() {
return eventSubject;
}
public String geteventContent() {
return eventContent;
}
}
public class Mail {
private String receiver;
private String subject;
private String content;
private String tail;
public Mail(EventTemplate et) {
this.tail = et.geteventContent();
this.subject = et.geteventSubject();
}
public String getReceiver() {
return receiver;
}
public void setReceiver(String receiver) {
this.receiver = receiver;
}
public String getSubject() {
return subject;
}
public void setSubject(String subject) {
this.subject = subject;
}
public void setContent(String content) {
this.content = content;
}
public String getContent() {
return content;
}
public String getTail() {
return tail;
}
public void setTail(String tail) {
this.tail = tail;
}
}
测试类
public class MainTest {
public static void main(String[] args) {
int i = 0;
int MAX_COUNT = 10;
EventTemplate et = new EventTemplate("9月份信用卡账单", "国庆抽奖活动...");
Mail mail = new Mail(et);
while (i < MAX_COUNT) {
// 以下是每封邮件不同的地方
mail.setContent(getRandString(5) + ",先生(女士):你的信用卡账单..."
+ mail.getTail());
mail.setReceiver(getRandString(5) + "@" + getRandString(8) + ".com");
// 然后发送邮件
sendMail(mail);
i++;
}
}
public static String getRandString(int maxLength) {
String source = "abcdefghijklmnopqrskuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ";
StringBuffer sb = new StringBuffer();
Random rand = new Random();
for (int i = 0; i < maxLength; i++) {
sb.append(source.charAt(rand.nextInt(source.length())));
}
return sb.toString();
}
public static void sendMail(Mail mail) {
System.out.println("标题:" + mail.getSubject() + "\t收件人:"
+ mail.getReceiver() + "\t内容:" + mail.getContent()
+ "\t....发送成功!");
}
}
运行结果
Exception in thread "main" java.lang.NullPointerException
at com.java.jikexueyuan.protomode.proto.Mail.clone(Mail.java:24)
at com.java.jikexueyuan.protomode.proto.MainTest.main(MainTest.java:17)
4. 原型模式与final字段的冲突
如果要克隆的类是一个final对象, 那么,调用的时候会报错。
因为final类指向的对象,地址是不能改变的,如果使用clone,等于在内存中又给该对象增加了一个地址的指针,会报错的。
5. 原型模式能够拷贝的内容
6. 原型模式适用场合:
1)复制对象的结构与数据
2)希望对目标对象的修改不影响既有的原型对象
3)创建对象成本较大的情况下
7. 关于深拷贝与浅拷贝
请参考