原型模式顾名思义通过一个接口实现快速创建对象
原型模式的本质:克隆生成对象。
克隆是手段,目的是生成新的对象实例。正是因为原型的目的是为了生成新的对象实例,原型模式通常是被归类为创建型的模式。
原型模式也可以用来解决“只知接口而不知实现的问题”,使用原型模式,可以出现一种独特的“接口造接口”的景象,这在面向接口编程中很有用。同样的功能也可以考虑使用工厂来实现。
另外,原型模式的重心还是在创建新的对象实例,至于创建出来的对象,其属性的值是否一定要和原型对象属性的值完全一样,这个并没有强制规定,只不过在目前大多数实现中,克隆出来的对象和原型对象的属性值是一样的。
也就是说,可以通过克隆来创造值不一样的实例,但是对象类型必须一样。可以有部分甚至是全部的属性的值不一样,可以有选择性地克隆,就当是标准原型模式的一个变形使用吧。
建议在以下情况时选用原型模式。
原型模式的优点
对客户端隐藏具体的实现类型
原型模式的客户端只知道原型接口的类型,并不知道具体的实现类型,从而减少了客户端对这些具体实现类型的依赖。
在运行时动态改变具体的实现类型
原型模式可以在运行期间,由客户来注册符合原型接口的实现类型,也可以动态地改变具体的实现类型,看起来接口没有任何变化,但其实运行的已经是另外一个类实例了。因为克隆一个原型就类似于实例化一个类。
原型模式的缺点
克隆接口
/*** @description:需要克隆方法实现此接口*/
public interface OrderApi {/*** 克隆方法* @return*/OrderApi clone();
}
实现类
/*** @description:个人订单*/
@Data
@ToString
public class PersonOrder implements OrderApi{/*** 用户名*/private String personName;/*** 产品id*/private Integer productId;private CompanyOrder companyOrder;/*** 克隆对象* @return*/@Overridepublic OrderApi clone() {PersonOrder personOrder = new PersonOrder();personOrder.setPersonName(this.personName);personOrder.setProductId(this.productId);//引用对象克隆personOrder.setCompanyOrder(this.companyOrder);return personOrder;}
}/*** @description:公司订单*/
@Data
@ToString
public class CompanyOrder implements OrderApi{/*** 公司名*/private String companyName;/*** 产品id*/private Integer productId;/*** 克隆对象* @return*/@Overridepublic OrderApi clone() {CompanyOrder companyOrder = new CompanyOrder();companyOrder.setCompanyName(this.companyName);companyOrder.setProductId(this.productId);return companyOrder;}
}
测试类
/*** @description:测试类*/
public class Client {public static void main(String[] args) {PersonOrder personOrder = new PersonOrder();personOrder.setPersonName("张三");personOrder.setProductId(10001);CompanyOrder companyOrder = new CompanyOrder();companyOrder.setCompanyName("字节跳动");companyOrder.setProductId(10002);personOrder.setCompanyOrder(companyOrder);System.out.println("原对象:"+personOrder);//自己实现克隆OrderApi personClone = personOrder.clone();System.out.println("克隆对象:"+personClone);//修改原对象中引用对象的值companyOrder.setCompanyName("88888888888");System.out.println("克隆对象:"+personClone);}
}
结果

实现深度克隆只需克隆引用对象时,调用它自己的克隆方法,一直传递下去
/*** 克隆对象* @return*/
@Override
public OrderApi clone() {PersonOrder personOrder = new PersonOrder();personOrder.setPersonName(this.personName);personOrder.setProductId(this.productId);personOrder.setCompanyOrder((CompanyOrder) this.companyOrder.clone());return personOrder;
}
结果

java提供的克隆
1、浅克隆:对当前对象进行克隆,并克隆该对象所包含的8种基本数据类型和String类型属性(拷贝一份该对象并重新分配内存,即产生了新的对象);但如果被克隆的对象中包含除8中数据类型和String类型外的其他类型的属性,浅克隆并不会克隆这些属性(即不会为这些属性分配内存,而是引用原来对象中的属性)
2、深克隆:深拷贝会完全复制整个对象,包括这个对象所包含的内部对象。
1.实现Cloneable接口
2.重写clone方法
/*** @description:个人订单*/
@Data
@ToString
public class PersonOrder2 implements Cloneable{/*** 用户名*/private String personName;/*** 产品id*/private Integer productId;private CompanyOrder2 companyOrder2;@Overrideprotected Object clone() throws CloneNotSupportedException {return super.clone();}
}@Data
@ToString
public class CompanyOrder2 implements Cloneable{/*** 公司名*/private String companyName;/*** 产品id*/private Integer productId;@Overrideprotected Object clone() throws CloneNotSupportedException {return super.clone();}
}
测试
/*** @description:测试类*/
public class Client {public static void main(String[] args) throws CloneNotSupportedException {CompanyOrder2 companyOrder2 = new CompanyOrder2();companyOrder2.setCompanyName("字节跳动");companyOrder2.setProductId(10002);PersonOrder2 personOrder2 = new PersonOrder2();personOrder2.setPersonName("张三");personOrder2.setProductId(10001);personOrder2.setCompanyOrder2(companyOrder2);System.out.println("原对象:"+personOrder2);//java浅克隆Object clone = personOrder2.clone();System.out.println("克隆对象:"+clone);companyOrder2.setCompanyName("9999999999");System.out.println("克隆对象:"+clone);}
}
结果

可以看到,虽然克隆出来了一个一模一样的对象,但是修改原对象的属性,克隆的对象属性也改变了,这就是浅克隆,原对象和克隆对象的引用属性共同指向一个内存地址
实现深度克隆只需克隆引用对象时,调用它自己的克隆方法,一直传递下去,原理和上面的原型实现深克隆一样
@Override
protected Object clone() throws CloneNotSupportedException {PersonOrder2 clone = (PersonOrder2) super.clone();clone.setCompanyOrder2((CompanyOrder2) this.companyOrder2.clone());return clone;
}
结果
