• 原型模式:

  原型模式,是指基于一个已经给定的对象,通过拷贝的方式,创建一个新的对象,这个给定对象,就是“原型”。

  在 Java 中,原型模式体现为 Object 的 clone() 方法。

  所有类都可以通过实现 Cloneable 接口,以及重写 clone() 方法,来实现原型模式。

  

  • 代码:
@Data
@Builder
@NoArgsConstructor
@AllArgsConstructor
public class Liability implements Cloneable {private String code;
private String name;
private String category;
private boolean isMajor; @Override
protected Liability clone() throws CloneNotSupportedException {
return (Liability) super.clone();
}
}
@Data
@Builder
public class PolicyShallowClone implements Cloneable { private String code;
private int applicantAge;
private Liability liability;
private List<String> specialDescriptions; @Override
public PolicyShallowClone clone() throws CloneNotSupportedException {
return (PolicyShallowClone) super.clone();
}
}
  • 缩减 clone() 方法的返回类型:

  自 JDK1.5 开始,Java 引进了一个新的特性:协变返回类型(covariant return type)。

  即:覆盖方法的返回类型,可以是被覆盖方法的返回类型的子类。

  所以需要在 clone() 方法内部进行强转。

  这体现了一条通则:永远不要让客户去做任何类库能够替客户完成的事情。

  • clone() 是一种浅度复制(Shallow Copy):
    @Test
void testPolicy1() throws Exception {
// Build original policy
Liability liability = new Liability.LiabilityBuilder().code("0001").name("Liability").category("XPXA").build();
String specialDescription1 = "text1";
String specialDescription2 = "text2";
List<String> specialDescriptions = new ArrayList<>(Arrays.asList(specialDescription1, specialDescription2));
PolicyShallowClone policyA = PolicyShallowClone.builder().specialDescriptions(specialDescriptions).liability(liability).code("code001").applicantAge(18).build();
// Call clone
PolicyShallowClone policyB = policyA.clone();
Assertions.assertSame(policyA.getCode(), policyB.getCode());
Assertions.assertEquals(policyA.getCode(), policyB.getCode());
// Assert shallow clone
policyA.getSpecialDescriptions().add("text3");
Assertions.assertSame(policyA.getLiability(), policyB.getLiability());
Assertions.assertTrue(policyA.getSpecialDescriptions().size() == policyB.getSpecialDescriptions().size());
}
  • 编写一个优秀的 clone() 方法:

  克隆对象的数据来源,必须来自于 clone() 方法,所以永远在方法内部调用 super.clone() 方法。

  所有的父类必须很好地实现了 clone() 方法。

  如果当前类包含的域引用了可变对象,需要递归地调用 clone() 方法。

  如果在线程安全的类中实现 Cloneable 接口,clone() 方法必须得到很好的同步。

  • 一个深度复制的 clone() 方法:
@Data
@Builder
public class PolicyDeepClone implements Cloneable { private String code;
private int applicantAge;
private Liability liability;
private List<String> specialDescriptions; @Override
public PolicyDeepClone clone() throws CloneNotSupportedException {
PolicyDeepClone clone = (PolicyDeepClone) super.clone();
clone.specialDescriptions = new ArrayList<>(this.specialDescriptions);
clone.liability = this.liability.clone();
return clone;
}
}
  • 深度复制的测试:
    @Test
void testPolicy2() throws Exception {
// Build original policy
Liability liability = new Liability.LiabilityBuilder().code("0001").name("Liability").category("XPXA").build();
String specialDescription1 = "text1";
String specialDescription2 = "text2";
List<String> specialDescriptions = new ArrayList<>(Arrays.asList(specialDescription1, specialDescription2));
PolicyDeepClone policyA = PolicyDeepClone.builder().specialDescriptions(specialDescriptions).liability(liability).code("code001").applicantAge(18).build();
// Call clone
PolicyDeepClone policyB = policyA.clone();
// Assert deep clone
policyA.getSpecialDescriptions().add("text3");
Assertions.assertNotSame(policyA.getLiability(), policyB.getLiability());
Assertions.assertFalse(policyA.getSpecialDescriptions().size() == policyB.getSpecialDescriptions().size());
}
  • 有必要这么复杂吗?

  从上述的介绍,我们不难发现,要完成一个优秀的 clone() 方法,存在诸多限制。

  并且,当我们实现了 clone() 方法,在编译器中,还会看到一条 Blocker 级别的 Sonar 警告:

  Remove this "clone" implementation; use a copy constructor or copy factory instead.

  它推荐的是一个拷贝构造器和拷贝工厂。

  • 拷贝构造器(Copy constructor)
@Data
@Builder
public final class PolicyCopyConstructor { private String code;
private int applicantAge;
private Liability liability;
private List<String> specialDescriptions; public PolicyCopyConstructor(PolicyCopyConstructor policy) {
this.code = policy.code;
this.applicantAge = policy.applicantAge;
this.liability = policy.liability;
this.specialDescriptions = policy.specialDescriptions;
}
}

  显然,这是一个浅度复制的实现,如果需要深度复制,需要深一步挖掘,这里不详述。

  • 拷贝工厂(Copy factory):
@Data
public final class PolicyCopyFactory { private String code;
private int applicantAge;
private Liability liability;
private List<String> specialDescriptions; public static PolicyCopyFactory newInstance(PolicyCopyFactory policy) {
PolicyCopyFactory copyPolicy = new PolicyCopyFactory();
copyPolicy.setCode(policy.getCode());
copyPolicy.setApplicantAge(policy.getApplicantAge());
copyPolicy.setLiability(policy.getLiability());
copyPolicy.setSpecialDescriptions(policy.getSpecialDescriptions());
return copyPolicy;
}
}

  拷贝工厂本质上使我们之前提到过的静态工厂的一种变形。

  在这里,这也是浅度复制的实现。

  • Copy constructor & Copy factory 的优势:
  1. 不依赖于某一种带有风险的,语言之外的对象创建机制(clone 是 native 方法)。
  2. 不会与 final 域的正常使用发生冲突(clone 架构与引用可变对象的 final 域的正常使用是不兼容的)。
  3. 不会抛出受检异常。
  4. 不需要类型转换。
  • 《Effective Java》 第11条:谨慎地覆盖 clone

  鉴于 clone() 方法存在这么多限制,《Effective Java》明确指出:

  除了拷贝数组,其他任何情况都不应该去覆盖 clone() 方法,也不该去调用它。

  • 关于深复制:

  这篇文章 第004弹:几种通用的深度复制的方法 介绍了几种深复制的通用方法。

设计模式(五)原型模式 Prototype的更多相关文章

  1. 设计模式五: 原型模式(Prototype)

    简介 原型模式是属于创建型模式的一种,是通过拷贝原型对象来创建新的对象. 万能的Java超类Object提供了clone()方法来实现对象的拷贝. 可以在以下场景中使用原型模式: 构造函数创建对象成本 ...

  2. 乐在其中设计模式(C#) - 原型模式(Prototype Pattern)

    原文:乐在其中设计模式(C#) - 原型模式(Prototype Pattern) [索引页][源码下载] 乐在其中设计模式(C#) - 原型模式(Prototype Pattern) 作者:weba ...

  3. 二十四种设计模式:原型模式(Prototype Pattern)

    原型模式(Prototype Pattern) 介绍用原型实例指定创建对象的种类,并且通过拷贝这个原型来创建新的对象.示例有一个Message实体类,现在要克隆它. MessageModel usin ...

  4. [设计模式] 4 原型模式 prototype

    设计模式:可复用面向对象软件的基础>(DP)本文介绍原型模式和模板方法模式的实现.首先介绍原型模式,然后引出模板方法模式. DP书上的定义为:用原型实例指定创建对象的种类,并且通过拷贝这些原型创 ...

  5. 设计模式 笔记 原型模式 prototype

    //---------------------------15/04/07---------------------------- //prototype 原型模式--对象创建型模式 /* 1:意图: ...

  6. python 设计模式之原型模式 Prototype Pattern

    #引入 例子1: 孙悟空拔下一嘬猴毛,轻轻一吹就会变出好多的孙悟空来. 例子2:寄个快递下面是一个邮寄快递的场景:“给我寄个快递.”顾客说.“寄往什么地方?寄给……?”你问.“和上次差不多一样,只是邮 ...

  7. 【UE4 设计模式】原型模式 Prototype Pattern

    概述 描述 使用原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象.如孙悟空猴毛分身.鸣人影之分身.剑光分化.无限剑制 原型模式是一种创建型设计模式,允许一个对象再创建另外一个可定制的对象, ...

  8. 【设计模式】—— 原型模式Prototype

    前言:[模式总览]——————————by xingoo 模式意图 由于有些时候,需要在运行时指定对象时哪个类的实例,此时用工厂模式就有些力不从心了.通过原型模式就可以通过拷贝函数clone一个原有的 ...

  9. 创建型设计模式之原型模式(Prototype)

    结构   意图 用原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象. 适用性 当要实例化的类是在运行时刻指定时,例如,通过动态装载:或者 为了避免创建一个与产品类层次平行的工厂类层次时:或 ...

  10. 设计模式之原型模式(prototype)

    原理:拷贝自身对象实际上就是调用的拷贝构造函数,注意事项是这里的拷贝是深拷贝,即需要拷贝指针所指的内容 #include <stdio.h> #include <memory> ...

随机推荐

  1. 【Python图像特征的音乐序列生成】生成伴奏旋律(附部分代码)

    做了半天做的都是一些细枝末节的东西,嗨呀. 伴奏旋律是Ukulele和弦,MIDI发音乐器是Guitar.在弹唱的时候,Ukulele和弦就是伴奏. 我们以创建<成都>伴奏为例: 节奏型: ...

  2. 不写画面的网页程序设计,Web API、Web Service、WCF Service

    客户有一个系统,经常要连上我方,查询数据 以前的作法是给对方一个账号,让他可以连上我们的DB来查询. 所以,早期的同仁,真的给他们DB链接字符串 客户的Windows程序.网站就真的靠这么危险的方式, ...

  3. java面试题(杨晓峰)---第四讲强引用、软引用、弱引用、幻想引用有什么区别?

    在java语言中,除了原始数据类型的变量,其他所有都是所谓的引用类型,指向各种不同的对象,理解引用对于掌握java对象生命周期和JVM内部相关机制非常有帮助. 今天问题:强引用.软引用.弱引用.幻想引 ...

  4. ABAP和Java的单元测试Unit Test

    ABAP ABAP class单元测试的执行入口,CLASS_SETUP, 是硬编码在单元测试框架实现CL_AUNIT_TEST_CLASS里的. 待执行的单元测试方法通过CL_AUNIT_TEST_ ...

  5. 使用Process组件访问本地进程

    实现效果; 知识运用: Process组件的StartInfo属性 //获取或设置要传递给Process的Start方法的属性 public ProcessStartInfo StartInfo {g ...

  6. Bootstrap历练实例:下拉菜单插件方法的使用

    <!DOCTYPE html><html><head><meta http-equiv="Content-Type" content=&q ...

  7. Bootstrap历练实例:标签页内的下拉菜单

    <!DOCTYPE html><html><head><meta http-equiv="Content-Type" content=&q ...

  8. Unicode与ASCiI之间有什么区别?java当中的转义字符 Character类的使用 String类的使用

    ASCII码 称为 美国标准信息交换码 (American standard code of Information Interchange) 其中一共有多少个码?2的7次幂 128个 Unicode ...

  9. c#—OpenFileDialog

    OpenFileDialog是什么? OpenFileDialog是一个类,实例化此类可以设置弹出一个文件对话框.比如:我们发邮件时需要上传附件的时候,就会弹出一个让我们选择文件的对话框, 我们可以根 ...

  10. 学习笔记(五): Feature Crosses

    目录 Feature Crosses Encoding Nonlinearity Kinds of feature crosses Glossay Crossing One-Hot Vectors P ...