原型
用原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象。
原型模式,即Prototype,是指创建新对象的时候,根据现有的一个原型来创建。
我们举个例子:如果我们已经有了一个String[]
数组,想再创建一个一模一样的String[]
数组,怎么写?
实际上创建过程很简单,就是把现有数组的元素复制到新数组。如果我们把这个创建过程封装一下,就成了原型模式。用代码实现如下:
1 2 3 4
| String[] original = { "Apple", "Pear", "Banana" };
String[] copy = Arrays.copyOf(original, original.length);
|
对于普通类,我们如何实现原型拷贝?Java的Object
提供了一个clone()
方法,它的意图就是复制一个新的对象出来,我们需要实现一个Cloneable
接口来标识一个对象是“可复制”的:
1 2 3 4 5 6 7 8 9 10 11 12 13 14
| public class Employee implements Cloneable { private int id; private String name; private int score;
public Object clone() { Employee employee = new Employee(); employee.id = this.id; employee.name = this.name; employee.score = this.score; return employee; } }
|
使用的时候,因为clone()
的方法签名是定义在Object
中,返回类型也是Object
,所以要强制转型,比较麻烦:
1 2 3 4 5 6 7 8 9
| Employee employee = new Employee(); employee.setId(123); employee.setName("Bob"); employee.setScore(88);
Employee employee2 = (Employee) employee.clone(); System.out.println(employee); System.out.println(employee2); System.out.println(employee == employee2);
|
实际上,使用原型模式更好的方式是定义一个copy()
方法,返回明确的类型:
1 2 3 4 5 6 7 8 9 10 11 12 13
| public class Employee { private int id; private String name; private int score;
public Employee copy() { Student employee = new Employee(); employee.id = this.id; employee.name = this.name; employee.score = this.score; return employee; } }
|
原型模式应用不是很广泛,因为很多实例会持有类似文件、Socket这样的资源,而这些资源是无法复制给另一个对象共享的,只有存储简单类型的“值”对象可以复制。