Android开发经验谈设计模式Android进阶之路

原型设计模式

2019-08-18  本文已影响5人  OneXzgj

1.定义:

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

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

2.使用场景

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.优缺点比价

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

上一篇 下一篇

猜你喜欢

热点阅读