原型模式
克隆羊问题
现在有一只羊tom,姓名为:tom,年龄为:1,颜色为:白色,请编写程序创建和tom羊属性完全相同的10只羊
传统方式
public class Sheep {
private String name;
private int age;
private String color;
public Sheep() {
}
public Sheep(String name, int age, String color) {
this.name = name;
this.age = age;
this.color = color;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
public String getColor() {
return color;
}
public void setColor(String color) {
this.color = color;
}
@Override
public String toString() {
return "Sheep{" +
"name='" + name + '\'' +
", age=" + age +
", color='" + color + '\'' +
'}';
}
}
public class Client {
public static void main(String[] args) {
// 传统的方法
Sheep sheep = new Sheep("tom", 1, "白色");
Sheep sheep1 = new Sheep(sheep.getName(), sheep.getAge(), sheep.getColor());
Sheep sheep2 = new Sheep(sheep.getName(), sheep.getAge(), sheep.getColor());
Sheep sheep3 = new Sheep(sheep.getName(), sheep.getAge(), sheep.getColor());
//....
System.out.println("sheep = " + sheep);
System.out.println("sheep1 = " + sheep1);
System.out.println("sheep2 = " + sheep2);
System.out.println("sheep3 = " + sheep3);
}
}
优点比较好理解,简单易操作
缺点在创建新的对象时,总是需要重新获取原始对象的属性,如果创建的对象比较复杂时,效率较低
总是需要重新初始化对象,而不是动态的获取对象运行的状态,不够灵活
思路:java中Object类时所有类的跟类,Object提供了一个clone()方法,该方法可以将一个java对象复制一份,但是需要实现clone的java类必须要实现一个接口Cloneable,该街廓表示该类能够复制且具有复制的能力=>原型模式
sheep
public class Sheep implements Cloneable {
private String name;
private int age;
private String color;
private String address = "蒙古羊";
public Sheep() {
}
public Sheep(String name, int age, String color) {
this.name = name;
this.age = age;
this.color = color;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
public String getColor() {
return color;
}
public void setColor(String color) {
this.color = color;
}
@Override
public String toString() {
return "Sheep{" +
"name='" + name + '\'' +
", age=" + age +
", color='" + color + '\'' +
", address='" + address + '\'' +
'}';
}
// 重写克隆的方法
// 克隆该实例 使用默认的克隆方法来完成
@Override
protected Object clone(){
Sheep sheep = null;
try {
sheep = (Sheep)super.clone();
} catch (CloneNotSupportedException e) {
System.out.println("e.getMessage() = " + e.getMessage());
}
return sheep;
}
}
public class Client {
public static void main(String[] args) {
System.out.println("原型模式完成对象的创建");
Sheep sheep = new Sheep("tom", 1, "白色");
Sheep sheep1 = (Sheep) sheep.clone();// 克隆
Sheep sheep2 = (Sheep) sheep.clone();// 克隆
Sheep sheep3 = (Sheep) sheep.clone();// 克隆
// ......
System.out.println("sheep = " + sheep);
System.out.println("sheep1 = " + sheep1);
System.out.println("sheep2 = " + sheep2);
System.out.println("sheep3 = " + sheep3);
}
}
原型模式在Spring框架中源码分析
1.spring中Bean的创建就是原型模式的应用
当Bean的作用域为prototype时
核心代码在AbstractBeanFactory中353行
spring中bean的作用域有singleton,prototype,request,session,global Session
深拷贝和浅拷贝
super.clone()默认是浅拷贝
深拷贝的概念
不仅要复制对象的所有基本数据类型的成员变量值,还要为所有引用数据类型的成员变量申请存储空间,并复制每个引用数据类型成员变量所引用的对象,知道该对象可达的所有对象。也就是说,对象进行深拷贝要对整个对象图进行拷贝,简单的说,深拷贝对引用数据类型的成员变量的对象图中所有的对象都开辟了内存空间;而浅拷贝只是传递地址指向,新的对象并没有对引用数据类型创建内存空间。
实现深拷贝的两种方法
1.重写clone方法