设计模式-原型模式

1 原型模式介绍

原型模式(Prototype)是一个创建型的模式,原型模式是有一个共有信息的样板实例,然后拷贝这个样板实例,而复制后的实例就是所谓的“原型”,这个原型是可以修改的。原型模式多用于创建复杂的或者构造耗时的实例,因为这种情况下, 复制一个已经存在的实例可以使程序运行更高效。

2 原型模式定义

用原型实例指定创建对象的种类,并通过拷贝这些原型创建新的对象

3 原型模式UML类图

原型模式-UML类图

在原型模式中有如下角色:

  • Client:客户端角色。
  • Prototype:抽象原型角色,抽象类或者接口,用来声明clone方法。
  • ConcretePrototype:具体的原型类,是客户端角色使用的对象,即被复制的对象。

4 原型模式的使用场景

  1. 类初始化需要消耗非常多的资源,这个资源包括数据、硬件资源等。通过原型拷贝避免这些消耗。
  2. 通过new产生一个独享需要非常频繁的数据准备或访问权限。
  3. 一个对象需要提供给其他对象访问,而且各个调用者可能都需要修改其值时,可以考虑使用原型模式拷贝多个对象供调用者使用,即保护性拷贝。

5 原型模式使用示例

下面我们模拟一个发送短信的例子,来看下原型模式的简单使用:
其中Message类扮演的是ConcretePrototypeCloneable就是Prototype
具体的原型类:

public class Message implements Cloneable{
    private String name;
    private double money;

    public Message() {
        System.out.println("执行构造函数Message");
    }

    public void setMessage(String name, double money) {
        this.name = name;
        this.money = money;
    }

    @NonNull
    @Override
    public Message clone() {
        Message message = null;
        try {
            message = (Message) super.clone();
        } catch (CloneNotSupportedException e) {
            e.printStackTrace();
        }

        return message;
    }

    public void sendMessage(){
        System.out.println(name +"您好:您今天消费了"+money+"元");
    }
}

Message 类实现了Cloneable接口,它是一个标识接口,表示这个对象是可拷贝的,只要重写clone方法就可以实现拷贝。

这里需要注意的是clone方法并不是Cloneable接口中的,而是Object中的方法。

下面我们看下客户端实现:

public class MbClient {
    public static void main(String[] args){
        Message message = new Message();
        message.setMessage("张三",100);

        Message message1 = message.clone();
        message1.setMessage("李四",200);

        Message message2 = message.clone();
        message2.setMessage("王五",300);

        message.sendMessage();
        message1.sendMessage();
        message2.sendMessage();
    }
}

我们可以看到李四和王五的消息是通过clone方法克隆的,而clone方法是不会执行构造函数的。输出的结果如下:

//执行构造函数Message
//张三您好:您今天消费了100.0元
//李四您好:您今天消费了200.0元
//王五您好:您今天消费了300.0元

如果Message 类没有实现了Cloneable接口,就去直接调用clone方法,就会抛出异常。输出结果如下:

执行构造函数Message
java.lang.CloneNotSupportedException: com.monkey.myapplication.mbdemo.Message
    at java.lang.Object.clone(Native Method)
    at com.monkey.myapplication.mbdemo.Message.clone(Message.java:27)
    at com.monkey.myapplication.mbdemo.MbClient.main(MbClient.java:13)
Exception in thread "main" java.lang.NullPointerException
    at com.monkey.myapplication.mbdemo.MbClient.main(MbClient.java:14)

6 浅拷贝和深拷贝

由于Object类提供的clone方法,不会拷贝对象中的内部数组和引用对象,所以就有了浅拷贝和深拷贝。

6.1 浅拷贝

我们继续使用发送短信的例子来看下浅拷贝,在Message类中有一个消费明细对象。

public class Message implements Cloneable{
    private String name; //姓名
    private ExpenseDetail detail;//消费明细

    public Message() {
        System.out.println("执行构造函数Message");
        detail = new ExpenseDetail();
    }

    public void setMessage(String name, String type,double money) {
        this.name = name;
        this.detail.setType(type);
        this.detail.setMoney(money);
    }

    @NonNull
    @Override
    public Message clone() {
        Message message = null;
        try {
            message = (Message) super.clone();
        } catch (CloneNotSupportedException e) {
            e.printStackTrace();
        }

        return message;
    }

    public void sendMessage(){
        System.out.println(name +"您好:您今天"+detail.getType()+"消费了"+detail.getMoney()+"元");
    }
}

消费明细类

public class ExpenseDetail{
    private String type;
    private double money;

    public String getType() {
        return type;
    }

    public void setType(String type) {
        this.type = type;
    }

    public double getMoney() {
        return money;
    }

    public void setMoney(double money) {
        this.money = money;
    }
}

客户端类

public class MbClient {
    public static void main(String[] args){
        Message message = new Message();
        message.setMessage("张三","吃饭",10);

        Message message1 = message.clone();
        message1.setMessage("李四","看电影",50);

        Message message2 = message.clone();
        message2.setMessage("王五","买书",100);

        message.sendMessage();
        message1.sendMessage();
        message2.sendMessage();
    }
}

输出的结果如下:

执行构造函数Message
张三您好:您今天买书消费了100.0元
李四您好:您今天买书消费了100.0元
王五您好:您今天买书消费了100.0元

我们可以看到所有人的消费明细居然都一样,这是因为Object类提供的clone方法,不会拷贝对象中的内部数组和引用对象,导致它们仍旧指向原来对象的内部元素地址,这种拷贝叫做浅拷贝。
由此而导致最后一次的值会覆盖前一次的值。

6.2 深拷贝

public class Message implements Cloneable{
    ...
    
    @NonNull
    @Override
    public Message clone() {
        Message message = null;
        try {
            message = (Message) super.clone();
            message.detail = this.detail.clone();//拷贝消费明细
        } catch (CloneNotSupportedException e) {
            e.printStackTrace();
        }

        return message;
    }
    ...

}

public class ExpenseDetail implements Cloneable{
    private String type;
    private double money;
    ...
    @NonNull
    @Override
    protected ExpenseDetail clone(){

        ExpenseDetail detail = null;
        try {
            detail = (ExpenseDetail) super.clone();
        } catch (CloneNotSupportedException e) {
            e.printStackTrace();
        }

        return detail;
    }
}

使用浅拷贝的客户端代码再次执行后,输出的结果如下:

执行构造函数Message
张三您好:您今天吃饭消费了10.0元
李四您好:您今天看电影消费了50.0元
王五您好:您今天买书消费了100.0元

拷贝Message对象的同时,也将它内部的引用对象ExpenseDetail进行拷贝,使得每个拷贝的对象之间无任何关联,都指向了自身对应的ExpenseDetail,这种拷贝就是深拷贝。

7 总结

原型模式本质上就是对象拷贝。使用原型模式可以解决构建复杂对象的资源消耗问题,能够在某些场景下提升创建对象的效率,还有一个特点就是保护性拷贝,如果我们操作时,不会对原有的对象造成影响。

优点:
原型模式是在内存中二进制流的拷贝,要比new一个对象的性能要好,特别是需要生产大量对象时。

缺点:
直接在内存中拷贝,构造函数是不会执行的,这样就减少了约束,既是优点也是缺点,在实际开发当中应注意这个问题。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 199,636评论 5 468
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 83,890评论 2 376
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 146,680评论 0 330
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 53,766评论 1 271
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 62,665评论 5 359
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,045评论 1 276
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,515评论 3 390
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,182评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,334评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,274评论 2 317
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,319评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,002评论 3 315
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,599评论 3 303
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,675评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,917评论 1 255
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,309评论 2 345
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 41,885评论 2 341

推荐阅读更多精彩内容