对于某些岗位来说,工作周报的内容会大同小异,如果用户每次都需要从空白的周报进行输入无疑会浪费用户很多的时间,如果周报能够按照用户的自定义来生成模板,或者从已有模板修改小部分得到新模板,这样用户的输入效率会大大提高。原型模式正是为解决这类问题而生。
原型模式:使用原型实例指定创建对象的种类,并且通过克隆这些原型创建新的对象。
原型模式是一种对象创建型模式。
原型模式的工作原理很简单,将一个原型对象传给那个要发动创建的对象,这个要发动创建的对象通过请求原型对象克隆自己来实现创建过程。原型模式是一种另类的创建型模式,创建克隆对象的工厂就是原型类自身,工厂方法由克隆方法实现。
通过克隆方法创建的对象是全新的对象,它们在内存中拥有新的地址,通常对克隆多产生的对象进行的修改不会对原型对象造成任何的影响,每一个克隆的对象都是相互独立的,通过不同的方式对克隆对象进行修改之后,可以得到一系列相似但不完全相同的对象。
Prototype
(抽象原型类):声明克隆方法的接口,是所有具体原型类的公共父类,可以是抽象类也可以是接口,还能是具体实现类ConcretePrototypr
(具体原型类):实现在抽象原型类中声明的克隆方法,在克隆方法中返回自己的一个克隆对象Client
(客户类):让一个原型对象克隆自身从而创建一个新的对象,在客户类中只需要直接实例化或通过工厂方法等方式创建一个原型对象,再通过调用该对象的克隆方法即可得到多个相同的对象。clone
的方法clone
clone
方法获取多个对象这里定义为接口:
interface Prototype
{
Prototype clone();
String getAttr();
void setAttr(String attr);
}
实现抽象原型接口,核心在于如何实现clone
,在Java中clone
通常有两种实现方式:
clone()
方法通用的克隆实现方法是在具体原型类的克隆方法中实例化一个与自身类型相同的对象并将其返回,并将相关的参数传入新创建的对象中,保证成员变量相同。
代码如下:
class ConcretePrototype implements Prototype
{
private String attr;
@Override
public String getAttr() {
return this.attr;
}
@Override
public void setAttr(String attr) {
this.attr = attr;
}
@Override
public Prototype clone()
{
Prototype = new ConcretePrototype();
prototype.setAttr(attr);
return prototype;
}
}
clone
java.lang.Object
提供了一个clone()
,可以将一个Java对象克隆一份,利用clone()
可以直接将对象克隆一份,但是必须实现Cloneable
接口,否则clone()
时会抛出CloneNotSupportedException
。
代码如下:
class ConcretePrototype implements Prototype,Cloneable
{
private String attr;
public String getAttr() {
return this.attr;
}
public void setAttr(String attr) {
this.attr = attr;
}
public Prototype clone()
{
Object object = null;
try
{
object = super.clone();
}
catch (Exception e)
{
e.printStacktrace();
}
return (Prototype)object;
}
}
一般而言,Java中的clone()
满足:
x
都有x.clone() != x
,也就是克隆的对象与原型对象不是同一个对象x
都有x.clone().getClass() == x.getClass()
,即克隆对象与原型对象的类型一样x
的equals()
定义恰当,那么x.clone().equals(x)
应该成立具体实现步骤如下:
clone()
,并声明为public
clone()
中调用super.clone()
Cloneable
接口客户类针对抽象原型类编程,通过实例化获取具体原型后,调用其中的clone
进行克隆:
public class Test
{
public static void main(String[] args) {
Prototype prototype1 = new ConcretePrototype();
prototype1.setAttr("test");
Prototype prototype2 = prototype1.clone();
System.out.println(prototype1.getAttr() == prototype2.getAttr());
System.out.println(prototype1 == prototype2);
}
}
开发一个工作周报系统,工作周报的内容都大同小异,只有一些小地方存在差异,但是系统每次默认创建的都是空白报表,用户不断复制粘贴来填写重复内容。使用原型模式对其进行优化,快速创建相同或类似的工作周报。
设计如下:
Object
)WeeklyLog
代码如下:
public class Test
{
public static void main(String[] args) {
WeeklyLog weeklyLog1 = new WeeklyLog();
weeklyLog1.setContent("content");
weeklyLog1.setName("Weekly log 1");
weeklyLog1.setDateTime(LocalDateTime.now());
System.out.println(weeklyLog1.getName());
System.out.println(weeklyLog1.getContent());
System.out.println(weeklyLog1.getDateTime());
WeeklyLog weeklyLog2 = weeklyLog1.clone();
weeklyLog2.setName("Weekly log 2");
System.out.println(weeklyLog2.getName());
System.out.println(weeklyLog2.getContent());
System.out.println(weeklyLog2.getDateTime());
}
}
class WeeklyLog implements Cloneable
{
private String name;
private LocalDateTime dateTime;
private String content;
//getter and setter
//...
public WeeklyLog clone()
{
Object obj = null;
try
{
obj = super.clone();
}
catch(Exception e)
{
e.printStackTrace();
}
return (WeeklyLog)obj;
}
}
一般来说,工作周报可能会携带附件,使用上面的原型模式来进行工作周报的复制没有问题,但是附件(一般是另一个类)不会进行复制。这是因为浅克隆与深克隆的原因,下面具体来看一下。
在浅克隆中,如果原型对象的成员变量是值类型,将复制一份给克隆对象。(在Java中)值类型包括:
int
double
byte
boolean
char
float
long
short
也就是这些类型的值都会完整复制一份给克隆对象,对于引用类型,则将引用对象的地址复制一份给克隆对象。(在Java中)引用类型就是除了基本类型之外的所有类型,常见的有:
对于引用类型,原型对象与克隆对象指向相同的内存地址,也就是其实并没有被复制,而是共享一份地址相同的值。
在Java中可以通过Object
的clone()
实现浅克隆,也就是上面例子的做法。
在深克隆中,无论变量是值类型还是引用类型都会完整复制一份给克隆对象。
在Java中实现深克隆可以通过序列化等方式实现。序列化就是将对象写到流的过程,写到流中的对象是原有对象的一个复制品,而原对象仍然存在于内存中。想要进行序列化必须实现Serializable
接口。
代码如下:
public class Test
{
public static void main(String[] args) {
WeeklyLog weeklyLog1 = new WeeklyLog();
WeeklyLog weeklyLog2 = null;
Attachement attachement = new Attachement();
weeklyLog1.setAttachement(attachement);
try
{
weeklyLog2 = weeklyLog1.deepClone();
}
catch(Exception e)
{
e.printStackTrace();
}
System.out.println(weeklyLog1 == weeklyLog2);
System.out.println(weeklyLog1.getAttachement() == weeklyLog2.getAttachement());
}
}
class Attachement implements Serializable
{
private String name;
//getter and setter
//...
}
class WeeklyLog implements Serializable
{
private String name;
private LocalDateTime dateTime;
private String content;
private Attachement attachement;
//getter and setter
//...
public WeeklyLog deepClone() throws IOException , ClassNotFoundException , OptionalDataException
{
ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream();
ObjectOutputStream objectOutputStream = new ObjectOutputStream(byteArrayOutputStream);
objectOutputStream.writeObject(this);
ByteArrayInputStream byteArrayInputStream = new ByteArrayInputStream(byteArrayOutputStream.toByteArray());
ObjectInputStream objectInputStream = new ObjectInputStream(byteArrayInputStream);
return (WeeklyLog)objectInputStream.readObject();
}
}
当然除了使用ByteArrayOutput/InputStream
以及ObjectInput/OutputStream
外,还可以利用以下工具类进行深克隆:
org.apache.commons.lang3.SerializationUtils.clone()
:需要实现Serializable
接口Gson
:无需实现Serializable
接口,toJson()+fromJson()
Jackson
:也是无需实现Serializable
接口,readValue()+writeValueAsString()
等原型管理器是将多个原型对象存储在一个集合中供客户端使用的专门负责克隆对象的工厂,其中定义了一个集合用于存储原型对象,如果需要某个原型对象的克隆,可以通过复制集合中对应的原型对象来获取。在原型管理器中针对抽象原型类进行编程。
结构图如下:
日常办公中会有许多公文需要创建,例如《可行性分析报告》,《立项建议书》,《软件需求规格说明书》,《项目进展报告》等,为了提高工作效率需要为各类公文创建模板,用户可以通过这些模板快速创建新的公文,这些公文模板进行统一的管理,系统根据用户的请求的不同生成不同的新公文。
首先是抽象原型以及具体原型的代码:
interface OfficialDocument extends Cloneable
{
OfficialDocument clone();
void display();
}
//可行性分析报告
class FAR implements OfficialDocument
{
public OfficialDocument clone()
{
OfficialDocument far = null;
try
{
far = (OfficialDocument)super.clone();
}
catch(Exception e)
{
e.printStackTrace();
}
return far;
}
public void display()
{
System.out.println("可行性分析报告");
}
}
//软件需求规格说明书
class SRS implements OfficialDocument
{
public OfficialDocument clone()
{
OfficialDocument srs = null;
try
{
srs = (OfficialDocument)super.clone();
}
catch(Exception e)
{
e.printStackTrace();
}
return srs;
}
public void display()
{
System.out.println("软件需求规格说明书");
}
}
接着是原型管理器的代码,使用枚举单例实现:
enum PrototypeManager
{
INSTANCE;
private Hashtable<String,OfficialDocument> hashtable = new Hashtable<>();
private PrototypeManager()
{
add("far",new FAR());
add("srs",new SRS());
}
public void add(String key,OfficialDocument document)
{
hashtable.put(key, document);
}
public OfficialDocument get(String key)
{
return ((OfficialDocument)hashtable.get(key)).clone();
}
}
测试代码:
public class Test
{
public static void main(String[] args) {
PrototypeManager manager = PrototypeManager.INSTANCE;
OfficialDocument document1,document2,document3,document4;
document1 = manager.get("far");
document1.display();
document2 = manager.get("far");
document2.display();
System.out.println(document1 == document2);
document3 = manager.get("srs");
document3.display();
document4 = manager.get("srs");
document4.display();
System.out.println(document3 == document4);
}
}
如果觉得文章好看,欢迎点赞。
同时欢迎关注微信公众号:氷泠之路。