020.原型模式

我们今天来考虑一下给用户邮箱发广告信这个模块是怎么开发的。既然是广告信,肯定需要一个模版,然后再从数据库中把客户的信息一个一个的取出,放到模版中生成一份完整的邮件,然后扔给发送机进行发送处理,我们来看类图:

在类图中AdvTemplate是广告信的模板,一般都是从数据库取出,生成一个BO或者是DTO,我们这里使用一个静态的值来做代表;Mail类是一个邮件类,发送机发送的就是这个类,我们先来看看我们的程序:

public class AdvTemplate {

    /**
     * 广告信名称
     */
    private String advSubject = "XX银行国庆信用卡抽奖活动";

    /**
     * 广告信内容
     */
    private String advContext = "国庆抽奖活动通知:只要刷卡就送你1百万!....";

    public String getAdvSubject() {
        return advSubject;
    }

    public String getAdvContext() {
        return advContext;
    }
}

public class Mail {

    /**
     * 收件人
     */
    private String receiver;

    /**
     * 主题
     */
    private String subject;

    /**
     * 称呼
     */
    private String appellation;

    /**
     * 邮件内容
     */
    private String context;

    /**
     * 邮件尾部信息
     */
    private String tail;

    public Mail(AdvTemplate advTemplate) {
        this.context = advTemplate.getAdvContext();
        this.subject = advTemplate.getAdvSubject();
    }

    public String getReceiver() {
        return receiver;
    }

    public void setReceiver(String receiver) {
        this.receiver = receiver;
    }

    public String getSubject() {
        return subject;
    }

    public void setSubject(String subject) {
        this.subject = subject;
    }

    public String getAppellation() {
        return appellation;
    }

    public void setAppellation(String appellation) {
        this.appellation = appellation;
    }

    public String getContext() {
        return context;
    }

    public void setContext(String context) {
        this.context = context;
    }

    public String getTail() {
        return tail;
    }

    public void setTail(String tail) {
        this.tail = tail;
    }
}

public class Client {

    /**
     * 发送邮件的数量
     */
    private static int maxCount = 6;

    public static void main(String[] args) {

        // 模拟发送邮件
        int i = 0;
        // 定义模板
        Mail mail = new Mail(new AdvTemplate());
        mail.setTail("XX银行版本所有");
        while (i < maxCount) {
            mail.setAppellation(getRandString(5) + " 先生/女士");
            mail.setReceiver(getRandString(5) + "@" + getRandString(8) + ".com");
            sendMail(mail);
            i++;
        }
    }

    /**
     * 发送邮件
     */
    public static void sendMail(Mail mail) {
        System.out.println(String.format("标题: %s, 收件人: %s ... 发送成功!", mail.getSubject(), mail.getReceiver()));
    }

    /**
     * 生成随机字符串
     * @param maxLength 字符串的最大长度
     * @return 生成的字符串
     */
    public static String getRandString(int maxLength) {
        String source = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ";
        StringBuilder sb = new StringBuilder();
        Random random = new Random();
        for (int i = 0; i < maxLength; i++) {
            sb.append(source.charAt(random.nextInt(source.length())));
        }
        return sb.toString();
    }

}

程序写出来了,我们考虑一个问题:发邮件可以使用多线程去发吗?当然是可以的,但是会有线程安全的问题,产生第一封邮件对象,放到线程1中运行,还没有发送出去;线程2也也启动了,直接就把邮件对象mail的收件人地址和称谓修改掉了,线程安全有多种解决办法,我们这里使用原型模式来解决这个问题,使用对象的拷贝功能来解决这个问题,类图稍作修改,如下图:

我们来看Mail类的改变:

public class Mail implements Cloneable {

    /**
     * 收件人
     */
    private String receiver;

    /**
     * 主题
     */
    private String subject;

    /**
     * 称呼
     */
    private String appellation;

    /**
     * 邮件内容
     */
    private String context;

    /**
     * 邮件尾部信息
     */
    private String tail;

    public Mail(AdvTemplate advTemplate) {
        this.context = advTemplate.getAdvContext();
        this.subject = advTemplate.getAdvSubject();
    }

    @Override
    protected Mail clone() throws CloneNotSupportedException {
        return (Mail)super.clone();
    }

    public String getReceiver() {
        return receiver;
    }

    public void setReceiver(String receiver) {
        this.receiver = receiver;
    }

    public String getSubject() {
        return subject;
    }

    public void setSubject(String subject) {
        this.subject = subject;
    }

    public String getAppellation() {
        return appellation;
    }

    public void setAppellation(String appellation) {
        this.appellation = appellation;
    }

    public String getContext() {
        return context;
    }

    public void setContext(String context) {
        this.context = context;
    }

    public String getTail() {
        return tail;
    }

    public void setTail(String tail) {
        this.tail = tail;
    }
}

Client类的改变:

public class Client {

    /**
     * 发送邮件的数量
     */
    private static int maxCount = 6;

    public static void main(String[] args) throws Exception {

        // 模拟发送邮件
        int i = 0;
        // 定义模板
        Mail mail = new Mail(new AdvTemplate());
        mail.setTail("XX银行版本所有");
        while (i < maxCount) {
            Mail cloneMail = mail.clone();
            cloneMail.setAppellation(getRandString(5) + " 先生/女士");
            cloneMail.setReceiver(getRandString(5) + "@" + getRandString(8) + ".com");
            sendMail(cloneMail);
            i++;
        }
    }

    /**
     * 发送邮件
     */
    public static void sendMail(Mail mail) {
        System.out.println(String.format("标题: %s, 收件人: %s ... 发送成功!", mail.getSubject(), mail.getReceiver()));
    }

    /**
     * 生成随机字符串
     * @param maxLength 字符串的最大长度
     * @return 生成的字符串
     */
    public static String getRandString(int maxLength) {
        String source = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ";
        StringBuilder sb = new StringBuilder();
        Random random = new Random();
        for (int i = 0; i < maxLength; i++) {
            sb.append(source.charAt(random.nextInt(source.length())));
        }
        return sb.toString();
    }

}

一样完成了电子广告信的发送功能,而且sendMail()即使是多线程也没有关系,mail.clone()这个方法把对象拷贝一份,产生一个新的对象,和原有对象一样,然后再修改细节的数据,如设置称谓,设置收件人地址等等。这种不通过new关键字来产生一个对象,而是通过对象拷贝来实现的模式就叫做原型模式,其通用类图如下:

这个模式的核心是一个clone()方法,通过这个方法进行对象的拷贝,Java提供了一个Cloneable接口来标示这个对象是可拷贝的,为什么说是“标示”呢?翻开JDK的帮助看看Cloneable是一个方法都没有的,这个接口只是一个标记作用,在JVM中具有这个标记的对象才有可能被拷贝,那怎么才能从“有可能被拷贝”转换为“可以被拷贝”呢?方法是覆盖clone()方法。

原型模式虽然很简单,但是在Java中使用原型模式也就是clone()方法还是有一些注意事项的:

  • 对象拷贝时,类的构造函数是不会被执行的,对象拷贝时确实构造函数没有被执行,这个从原理来讲也是可以讲得通的,Object类的clone()方法的原理是从推内存中以二进制流的方式进行拷贝,重新分配一个内存块,那构造函数没有被执行也是非常正常的了。

    public class CloneExample implements Cloneable {
    
        public CloneExample() {
            System.out.println("调用构造器...");
        }
    
        @Override
        protected CloneExample clone() throws CloneNotSupportedException {
            return (CloneExample)super.clone();
        }
    
        public static void main(String[] args) throws Exception {
    
            CloneExample ce1 = new CloneExample();
            CloneExample ce2 = ce1.clone();
            System.out.println(ce1);
            System.out.println(ce2);
    
        }
    }
    
  • 浅拷贝和深拷贝问题

    public class CloneExample2 implements Cloneable {
    
        private ArrayList<String> arrayList = new ArrayList<>();
    
        @Override
        protected CloneExample2 clone() throws CloneNotSupportedException {
            return (CloneExample2)super.clone();
        }
    
        public void setValue(String value) {
            arrayList.add(value);
        }
    
        public ArrayList<String> getValue() {
            return arrayList;
        }
    
        public static void main(String[] args) throws Exception {
            CloneExample2 ce1 = new CloneExample2();
            ce1.setValue("张三");
            CloneExample2 ce2 = ce1.clone();
            ce2.setValue("李四");
            System.out.println(ce1.getValue()); // 结果是: [张三, 李四]
        }
    
    }
    

    怎么会有李四呢?是因为Java做了一个偷懒的拷贝动作,Object类提供的方法clone() 只是拷贝本对象,其对象内部的数组、引用对象等都不拷贝,还是指向原生对象的内部元素地址,这种拷贝就叫做浅拷贝,确实是非常浅,两个对象共享了一个私有变量,你改我改大家都能改,是一个种非常不安全的方式,在实际项目中使用还是比较少的。你可能会比较奇怪,为什么在Mail那个类中就可以使用String类型,而不会产生由浅拷贝带来的问题呢?内部的数组和引用对象才不拷贝,其他的原始类型比如intlongString(Java就希望你把String认为是基本类型,String是没有clone()方法的)等都会被拷贝的。浅拷贝是有风险的,那怎么才能深入的拷贝呢?我们修改一下我们的程序:

    @Override
    protected CloneExample2 clone() throws CloneNotSupportedException {
      /*
      * 浅拷贝
      * return (CloneExample2)super.clone();
      */
        /*
         * 深拷贝
         */
      CloneExample2 ce = (CloneExample2)super.clone();
      ce.arrayList = (ArrayList<String>)arrayList.clone();
      return ce;
    }
    

    深拷贝还有一种实现方式就是通过自己写二进制流来操作对象,然后实现对象的深拷贝,深拷贝和浅拷贝建议不要混合使用,一个类中某些引用使用深拷贝,某些引用使用浅拷贝,这是一种非常差的设计,特别是是在涉及到类的继承,父类有几个引用的情况就非常的复杂,建议的方案深拷贝和浅拷贝分开实现。

  • 对象的clone()与对象内的final属性是冲突的

    public class CloneExample3 implements Cloneable {
    
        private final ArrayList<String> arrayList = new ArrayList<>();
    
        @Override
        protected CloneExample3 clone() throws CloneNotSupportedException {
            CloneExample3 ce = (CloneExample3)super.clone();
            ce.arrayList = (ArrayList<String>)arrayList.clone(); // 编译报错
            return ce;
        }
    
        public void setValue(String value) {
            arrayList.add(value);
        }
    
        public ArrayList<String> getValue() {
            return arrayList;
        }
    }
    

原型模式的适用场景:

  • 一是类初始化需要消化非常多的资源,这个资源包括数据、硬件资源等;
  • 二是通过new产生一个对象需要非常繁琐的数据准备或访问权限,则可以使用原型模式;
  • 三是一个对象需要提供给其他对象访问,而且各个调用者可能都需要修改其值时,可以考虑使用原型模式拷贝多个对象供调用者使用。在实际项目中,原型模式很少单独出现,一般是和工厂方法模式一起出现,通过clone()方法创建一个对象,然后由工厂方法提供给调用者。

本文原书:

《您的设计模式》 作者:CBF4LIFE

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

推荐阅读更多精彩内容

  • 原型模式定义:用原型实例指定创建对象的种类, 并且通过拷贝这些原型创建新的对象。 原型模式的核心是一个clone方...
    代码墨白阅读 89评论 0 0
  • 概念 在Java中实现原型模式十分简单,只需要实现Cloneable接口并重写clone()方法就可以了 Code...
    tanoak阅读 615评论 0 0
  • 1、原型模式 1、定义 拷贝一个对象创建新的对象 2、使用场景 类初始化需要消耗非常多的资源; 通过new需要非常...
    Dane_404阅读 138评论 0 0
  • 银行发广告信,为了提供个性化服务,发过去的邮件需要带上个人信息,如XX先生/小姐,又或者是电子账单,这就需要一个模...
    凉快先生阅读 250评论 0 1
  • 1大同小异的工作周报 Sunny软件公司一直使用自行开发的一套OA (Office Automatic,办公自动化...
    justCode_阅读 1,145评论 0 3