原型设计模式

作者: OneXzgj | 来源:发表于2019-08-18 11:33 被阅读5次

    1.定义:

    原型模式(Prototype Pattern)是用于创建重复的对象,同时又能保证性能。这种类型的设计模式属于创建型模式,它提供了一种创建对象的最佳方式。

    这种模式是实现了一个原型接口,该接口用于创建当前对象的克隆。当直接创建对象的代价比较大时,则采用这种模式。例如,一个对象需要在一个高代价的数据库操作之后被创建。我们可以缓存该对象,在下一个请求时返回它的克隆,在需要的时候更新数据库,以此来减少数据库调用。(即数据持久化)

    2.使用场景

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

    3.UML图示

    image.png

    4.简单示例

    以原型模式的核心是clone方法,通过该方法进行拷贝,这里举一个名片拷贝的例子。
    现在已经流行电子名片了,只要扫一下就可以将名片拷贝到自己的名片库中, 我们先实现名片类。

    public class BusinessCard implements Cloneable {
        
        private String name;
        private String company;
    
        public BusinessCard() {
            System.out.println("执行了构造方法BusinessCard构造方法");
        }
    
    
        @Override
        protected BusinessCard clone()  {
            
            BusinessCard card=null;
            try {
                card= (BusinessCard) super.clone();
                return card;
            } catch (CloneNotSupportedException e) {
                e.printStackTrace();
            }
            return null;
        }
    
        public String getName() {
            return name;
        }
        
        public String getCompany() {
            return company;
        }
    
        public void show(){
            System.out.println("BusinessCard{" +
                    "name='" + name + '\'' +
                    ", company='" + company + '\'' +
                    '}');
        }
    }
    
       public static void main(String[] args) {
    
            BusinessCard card=new BusinessCard();
            card.setName("张三");
            card.setCompany("阿里");
            
            BusinessCard clone = card.clone();
            clone.setName("李四");
            clone.setCompany("百度");
            
            BusinessCard clone2 = card.clone();
            clone2.setName("王五");
            clone2.setCompany("腾讯");
    
            card.show();
            clone.show();
            clone2.show();
          
    }
    

    运行结果


    image.png

    5.浅拷贝和深拷贝

    原型模式涉及到浅拷贝和深拷贝的知识点,为了更好的理解它们,还需要举一些例子。

    5.1实现浅拷贝

    上述的例子中,BusinessCard的字段都是String类型的,如果字段是引用的类型的,会出现什么情况呢

    package com.onexzgj.yuanxing;
    
    public class DeepBusinessCard implements Cloneable {
    
        private String name;
        private Company company =new Company();
    
        public DeepBusinessCard() {
            System.out.println("执行了构造方法BusinessCard构造方法");
        }
    
        @Override
        protected DeepBusinessCard clone()  {
    
            DeepBusinessCard card=null;
            try {
                card= (DeepBusinessCard) super.clone();
               
                return card;
            } catch (CloneNotSupportedException e) {
                e.printStackTrace();
            }
            return null;
        }
    
      ....
       public void setCompany(String name,String address) {
            this.company.setName(name);
            this.company.setAddress(address);
        }
    
        public void show(){
            System.out.println("BusinessCard{" +
                    "name='" + name + '\'' +
                    ", address='" + company.getAddress() + '\'' +"name"+company.getName()+
                    '}');
        }
    
    }
    
    public class Client {
        public static void main(String[] args) {
    
            DeepBusinessCard card=new DeepBusinessCard();
            card.setName("张三");
            card.setCompany("阿里","望京");
    
            DeepBusinessCard clone = card.clone();
            clone.setName("李四");
            clone.setCompany("百度","中关村");
    
            DeepBusinessCard clone2 = card.clone();
            clone2.setName("王五");
            clone2.setCompany("腾讯","西二旗");
    
            clone.show();
            card.show();
            clone2.show();
    
        }
    }
    

    运行结果

    image.png

    从结果可以看出company字段为最后设置的”腾讯”、”西二旗”。这是因为Object类提供的clone方法,不会拷贝对象中的内部数组和引用对象,导致它们仍旧指向原来对象的内部元素地址,这种拷贝叫做浅拷贝。

    5.2 实现深拷贝

    首先修改Company类实现Cloneable接口

    public class Company implements Cloneable{
        private String name;
        private String address;
        ...
        public Company clone(){
            Company company=null;
            try {
                company= (Company) super.clone();
            } catch (CloneNotSupportedException e) {
                e.printStackTrace();
            }
            return company;
        }
    }
    

    为了实现Company类能被拷贝,Company类也需要实现Cloneable接口并且覆写clone方法。接着修改DeepBusinessCard的clone方法:

    public class DeepBusinessCard implements Cloneable {
        private String name;
        private Company company = new Company();
        ...
        @Override
        public DeepBusinessCard clone() {
            DeepBusinessCard businessCard = null;
            try {
                businessCard = (DeepBusinessCard) super.clone();
                businessCard.company = this.company.clone();//1
    
            } catch (CloneNotSupportedException e) {
                e.printStackTrace();
            }
            return businessCard;
        }
      ...
    }
    

    运行结果

    image.png

    6.优缺点比价

    • 优点:
      1、性能提高。
      2、逃避构造函数的约束。

    • 缺点:
      1、配备克隆方法需要对类的功能进行通盘考虑,这对于全新的类不是很难,但对于已有的类不一定很容易,特别当一个类引用不支持串行化的间接对象,或者引用含有循环结构的时候。
      2、必须实现 Cloneable 接口。
      3、直接在内存中拷贝,构造函数是不会执行的,这样就减少了约束,这既是优点也是缺点,需要在实际应用中去考量。

    参考文章:
    刘望舒博客
    《Android源码设计模式解析与实战》

    相关文章

      网友评论

        本文标题:原型设计模式

        本文链接:https://www.haomeiwen.com/subject/hznksctx.html