原型模式:
通过new产生一个对象需要非常繁琐的数据准备或访问权限,则可以使用原型模式。原型模式就是java中的克隆技术,以某个对象为原型,复制出新的对象。显然,新的对象具备原型对象的特点。
原型模式优势:效率高(直接克隆,避免了重新执行构造过程步骤)。
克隆类似于new,但是不同于new。new创建新的对象属性采用的是默认值。克隆出的对象的属性完全和原型对象相同。并且克隆出的新对象改变不会影响原型对象。然后,再修改克隆对象的值。
原型模式实现:
Cloneable接口和clone方法。(Prototype模式实现起来最困难的地方就是内存复制操作,所幸在java中提供了clone方法替代我们做了绝大部分事情)
/**
* 猴子
*/
public class Monkey implements Cloneable{
private String name;
private Date birthday;
@Override
protected Object clone() throws CloneNotSupportedException {
Object obj = super.clone();//浅复制,直接调用object对象的clone方法,没有调用
Date对象的clone方法
return obj;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public Date getBirthday() {
return birthday;
}
public void setBirthday(Date birthday) {
this.birthday = birthday;
}
public Monkey(String name, Date birthday) {
super();
this.name = name;
this.birthday = birthday;
}
public Monkey() {
}
}
/**
*测试原型模式(浅复制)
*/
public class Client {
public static void main(String[] args) throws CloneNotSupportedException{
Date date = new Date(500000L);
Monkey dasheng = new Monkey("孙悟空",date);
Monkey monkey = (Monkey)dasheng.clone();//克隆出一个新猴子
System.out.println("1.名字:"+dasheng.getName()+"\t\t出生日期:"+dasheng.getBirthday());
System.out.println("------------------修改出生日期--------");
date.setTime(1234567890L);
System.out.println("2.名字:"+dasheng.getName()+"\t\t出生日期:"+dasheng.getBirthday());
monkey.setName("孙影---孙悟空的影子");
System.out.println("3.名字:"+monkey.getName()+"\t\t出生日期:"+monkey.getBirthday());
}
}
运行的结果如下:
由上面运行结果,可知,当修改出生日期date的值,孙悟空和孙影的出生日期的值都跟着改变了,这说明dasheng和monkey对象共同指向了同一个date对象,这就属于浅复制。
深复制就是当克隆时,把属性对象都重新分配属性对象,也就说克隆前对象和克隆后对象的属性分别指向了不同的地址空间。深复制实现如下:
/**
* 猴子
*/
public class Monkey implements Cloneable{
private String name;
private Date birthday;
@Override
protected Object clone() throws CloneNotSupportedException {
Object obj = super.clone();//直接调用object对象的clone方法
//添加如下代码实现深复制
Monkey monkey = (Monkey)obj;
monkey.birthday = (Date)this.birthday.clone();//把Date属性也进行复制
return monkey;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public Date getBirthday() {
return birthday;
}
public void setBirthday(Date birthday) {
this.birthday = birthday;
}
public Monkey(String name, Date birthday) {
super();
this.name = name;
this.birthday = birthday;
}
public Monkey() {
}
}
/**
*测试原型模式(深复制)
*/
public class Client {
public static void main(String[] args) throws CloneNotSupportedException{
Date date = new Date(500000L);
Monkey dasheng = new Monkey("孙悟空",date);
Monkey monkey = (Monkey)dasheng.clone();//克隆出一个新猴子
System.out.println("1.名字:"+dasheng.getName()+"\t\t出生日
期:"+dasheng.getBirthday());
System.out.println("------------------修改出生日期--------");
date.setTime(1234567890L);
System.out.println("2.名字:"+dasheng.getName()+"\t\t出生日
期:"+dasheng.getBirthday());
monkey.setName("孙影---孙悟空的影子");
System.out.println("3.名字:"+monkey.getName()+"\t\t出生日
期:"+monkey.getBirthday());
}
}
运行结果如下:
原型模式的注意事项:
1.使用原型模式复制对象不会调用类的构造方法。因为对象的复制是通过调用Object类的clone方法来完成的,它直接在内存中复制数据,因此不会调用到类的构造方法。不但构造方法中的代码不会执行,甚至连访问权限都对原型模式无效。
2.深拷贝与浅拷贝。Object类的clone方法只会拷贝对象中的基本的数据类型,对于数组、容器对象、引用对象等都不会拷贝,这就是浅拷贝。如果要实现深拷贝,必须将原型模式中的数组、容器对象、引用对象等另行拷贝。