个人博客

http://www.milovetingting.cn

原型模式

模式介绍

原型模式是一个创建型的模式。多用于创建复杂的或者构造耗时的实例,因为这种情况下,复制一个已经存在的实例,可使程序运行更高效。

模式定义

用原型实例指定创建对象的种类,并通过复制这些原型创建新的对象。

使用场景

  1. 类初始化需要消耗非常多的资源

  2. 通过new产生一个对象需要非常繁琐的数据准备或访问权限

  3. 一个对象需要提供给其它对象访问,而且各个调用者可能都需要修改值

通过实现Cloneable接口的原型模式在调用clone函数构造实例时,并不一定比通过new操作速度快,只有当通过new构造函数对象较为耗时或成本较高时,通过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
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
public class Document implements Cloneable {

private String mText;
private ArrayList<String> mImages = new ArrayList<>();

public String getmText() {
return mText;
}

public void setmText(String mText) {
this.mText = mText;
}

public List<String> getmImages() {
return mImages;
}

public void addImage(String image) {
mImages.add(image);
}

public void showDocument() {
System.out.println(this);
}

@Override
protected Object clone() throws CloneNotSupportedException {
Document doc = (Document) super.clone();
doc.mText = this.mText;
doc.mImages = this.mImages;
return doc;
}

@Override
public String toString() {
return "Document [mText=" + mText + ", mImages=" + mImages + "]";
}

}

调用

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
public static void main(String[] args) {
Document doc1 = new Document();
doc1.setmText("文档1");
doc1.addImage("图片1");
doc1.showDocument();
Document doc2;
try {
doc2 = (Document) doc1.clone();
doc2.setmText("文档2");
doc2.addImage("图片2");
doc2.showDocument();
doc.showDocument();
} catch (CloneNotSupportedException e) {
e.printStackTrace();
}

}

创建文档1,然后显示文档1的内容。然后拷贝了文档1,修改了文档1的内容,然后依次显示文档2,文档1

输出结果

1
2
3
Document [mText=文档1, mImages=[图片1]]
Document [mText=文档2, mImages=[图片1, 图片2]]
Document [mText=文档1, mImages=[图片1, 图片2]]

可以看到,修改拷贝后的文档2的Text,文档1没有受影响,但是修改文档2的Images,文档1也被修改了。

下面演示深拷贝

只需要修改clone方法

1
2
3
4
5
6
7
8
9
@SuppressWarnings("unchecked")
@Override
protected Object clone() throws CloneNotSupportedException {
Document doc = (Document) super.clone();
doc.mText = this.mText;
//doc.mImages = this.mImages;
doc.mImages = (ArrayList<String>) this.mImages.clone();
return doc;
}

再次运行后的输出结果:

1
2
3
Document [mText=文档1, mImages=[图片1]]
Document [mText=文档2, mImages=[图片1, 图片2]]
Document [mText=文档1, mImages=[图片1]]

修改文档2的Images并没有影响到文档1。

小结

原型模式本质上就是对象拷贝,容易出现的问题是深拷贝、浅拷贝。使用原型模式可以解决构建复杂对象的资源消耗问题,能够在某些场景下提升创建对象的效率。还有一个重要用途是保护性拷贝,也就是某个对象对外可能是只读的,为了防止外部对这个只读对象修改,可以通过返回一个对象拷贝的形式来实现只读的限制。