什么是原型模式
原型模式是一种创建型模式,它采取复制原型对象的方法来创建对象的实例。使用原型模式创建的实例,具有与原型一样的数据。
原型模式的特点
1、由原型对象自身创建目标对象。也就是说,对象创建这一动作发自原型对象本身。
2、目标对象是原型对象的一个克隆。也就是说,通过原型模式创建的对象,不仅仅与原型对象具有相同的结构,还与原型对象具有相同的值。
3、根据对象克隆深度层次的不同,有浅度克隆与深度克隆。
原型模式的使用场景
一、在创建对象的时候,我们不只是希望被创建的对象继承基类的基本结构,还希望继承原型对象的数据。
二、希望对目标对象的修改不影响既有的原型对象(深度克隆的时候可以完全互不影响)。
三、隐藏克隆操作的细节。很多时候,对对象本身的克隆需要涉及到类本身的数据细节。
原型模式的实现
创建原型模式只需要实现Cloneable。也就是说该类被声明可以被克隆
//实现接口
public class Person implements Cloneable{
private String name;
private int age;
private String sex;
private List<String> friends;
................省略set get方法
//这是克隆核心,浅度的克隆
public Person clone(){
try {
return (Person) super.clone();//这是克隆出来的person
} catch (CloneNotSupportedException e) {
e.printStackTrace();
return null;
}
}
}
Person person1 = new Person();
person1.setName("张三");
person1.setAge(25);
person1.setSex("男");
List<String> friends = new ArrayList<String>();
friends.add("he");
friends.add("jing");
friends.add("feng");
friends.add("lai");
person1.setFriends(friends);
//Person person2 = person1;这里person2在内存中并没创建一个新的对象,只是引用指向了原有person
//克隆原对象。这里引用是指向的是在内存中重新创建的一个对象,只是它的数据和原来的对象一样。
Person person2 = person1.clone();
person1.setName("zs");//这里修改person1对象的内容,不影响person2对象的内容
System.out.println(person1.getName());//运行后结果 zs
System.out.println(person1.getAge());
System.out.println(person1.getSex());
System.out.println(person2.getName());//运行后结果 张三
System.out.println(person2.getAge());
System.out.println(person2.getSex());
运行后结果
上面都是浅度克隆结果,引用类型的属性需要深度处理。
例如:friends属性,克隆体只是克隆了它的引用
//深度克隆核心
public Person clone(){
try {
//深度克隆
Person person = (Person) super.clone();//这是克隆出来的person
List<String> friendList = new ArrayList<>();
for(String friend : this.getFriends()){//取得原person的数据
friendList.add(friend);
}
person.setFriends(friendList);//原克隆数据复制到克隆person内
return person;//返回出克隆体
} catch (CloneNotSupportedException e) {
e.printStackTrace();
return null;
}
}
Person person3 = person1.clone();
System.out.println(person1.getFriends());
System.out.println(person3.getFriends());
friends.add("fang");
friends.add("wang");
friends.add("ni");
person1.setFriends(friends);
//这里的friends属性克隆的只是它的一个引用
//所以当person1改变之后,person3的friend也会改变
System.out.println(person1.getFriends());
System.out.println(person3.getFriends());
运行结果