设计模式---原型模式

目录

简介

1. 模式动机

2. 深克隆和浅克隆

3. 适用场景

实现

优缺点

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

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

简介

1. 模式动机

  \bullet 在面向对象系统中,使用原型模式来复制一个对象自身,从而克隆出多个与原型对象一模一样的对象。

  \bullet 在软件系统中,有些对象的创建过程比较复杂,而且有时候需要频繁创建,原型模式通过给出一个原型对象来指明要创建的对象类型,然后用复制这个原型对象的办法创建更多同类型的对象,这就是原型模式的意图所在。

  \bullet 原型模式将克隆过程委派给被克隆的实际对象。模式为所有支持克隆的对象声明一个通用的接口,该接口让你能够克隆对象,同时又无需将代码和对象所属类耦合。通常情况下,这样的接口中仅包含一个克隆方法。

  \bullet 所有的类对克隆方法的实现都非常相似。该方法会创建一个当前类的对象,然后将原始对象所有的成员变量值复制到新建的类中。甚至可以复制私有成员变量,因为绝大部分编程语言都允许对象访问其同类对象的私有成员变量。

2. 深克隆和浅克隆

 通常情况下,一个类包含一些成员对象,在使用原型模式克隆对象时,根据其成员对象是否也克隆,原型模式可以分为两种形式:深克隆和浅克隆

  \bullet 浅克隆:创建一个新对象,新对象的属性和原来对象完全相同,对于那些引用其它对象的属性,仍指向原有属性所指向的对象的内存地址。

  \bullet 深克隆:创建一个新对象,属性中引用的其他对象也会被克隆,不再指向原有对象地址。

3. 适用场景

  \bullet 对象之间相同或相似,即只是个别的几个属性不同的时候。

  \bullet 创建对象成本较大,例如初始化时间长,占用 CPU 太多,或者占用网络资源太多等,需要优化资源。

  \bullet 创建一个对象需要繁琐的数据准备或访问权限等,需要提高性能或者提高安全性。

  \bullet 系统中大量使用该类对象,且各个调用者都需要给它的属性重新赋值。

实现

我们将创建一抽象类 Shape 和扩展 Shape 类的实体类。下一步是定义类 ShapeCache,该类把 shape 对象存储在一个 Hashtable 中,并在请求的时候返回它们的克隆。

PrototypePatternDemo 类使用 ShapeCache 类来获取 Shape 对象。

 步骤1:创建一个实现 Cloneable 接口的抽象类。

public abstract class Shape implements Cloneable{

    private String id;
    protected String type;

    abstract void draw();

    public String getType(){
        return type;
    }

    public String getId(){
        return id;
    }

    public void setId(String id){
        this.id = id;
    }

    public Object clone(){
        Object clone = null;
        try{
            clone = super.clone();
        }catch(CloneNotSupportException e) {
            e.printStackTrace();
        }
        return clone;
    }
}

步骤2:创建扩展了上面抽象类的实体类

public class Rectangle extends Shape{
    
    public Rectangle(){
        type = "Rectangle";
    }

    // Shape 里的 draw() 是被 abstract 关键字修饰的,所以子类必须实现它
    @Override
    public void draw(){
        System.out.println("Rectangle");
    }
}
public class Square extends Shape{
    
    public Rectangle(){
        type = "Square ";
    }

    // Shape 里的 draw() 是被 abstract 关键字修饰的,所以子类必须实现它
    @Override
    public void draw(){
        System.out.println("Square ");
    }
}
public class Circle extends Shape{
    
    public Rectangle(){
        type = "Circle ";
    }

    // Shape 里的 draw() 是被 abstract 关键字修饰的,所以子类必须实现它
    @Override
    public void draw(){
        System.out.println("Circle ");
    }
}

步骤3:创建 ShapeCache 类,并把它们存储在一个 Hashtable 中。

import java.util.Hashtable;

public class ShapeCache{
    
    private static Hashtable<String, Shape> shapeMap = new Hashtable<String, Shape>();
    
    public static Shape getShape(String shapeId){
        Shape cachedShape = shapeMap.get(shapeId);
        return (Shape)cacheShape.clone();
    }

    // 对每种形状都运行数据库查询,并创建该形状
    // shapeMap.put(shapeKey, shapeValue);
    // 例如,我们要添加三种形状
    public static void loadCache(){
        Circle circle = new Circle();
        circle.setId("1");
        shapeMap.put(circle.getId(), circle);

        Square square = new Square();
        square.setId("2");
        shapeMap.put(square.getId(), square);

        Rectangle rectangle = new Rectangle ();
        rectangle .setId("3");
        shapeMap.put(rectangle .getId(), rectangle );

    }
}

步骤4:PrototypePatternDemo 使用 ShapeCache 类来获取存储在 Hashtable 中形状的克隆。

public class PrototypePatternDemo{
    
    public static void main(Strign[] args){
        ShapeCache.loadCache();

        //cloneShape == ShapeCache.getShape(id) 根据 id 去克隆对应的 Shape 
        Shape cloneShape = (Shape) ShapeCache.getShape("1");
        System.out.println("Shape" + cloneShape.getType());

        Shape cloneShape2 = (Shape) ShapeCache.getShape("2");
        System.out.println("Shape" + cloneShape2.getType());

        Shape cloneShape3 = (Shape) ShapeCache.getShape("3");
        System.out.println("Shape" + cloneShape3.getType());
        
    }
}

步骤5:输出结果

Shape : Circle
Shape : Square
Shape : Rectangle

注意:与通过对一个类进行实例化来构造新对象不同的是,原型模式是通过拷贝一个现有对象生成新对象的。浅拷贝实现 Cloneable 接口,深拷贝则是通过实现 Serializable 读取二进制流。

优缺点

优点

  \bullet 当创建新的对象实例较为复杂时,使用原型模式可以简化对象的创建过程,通过一个已有实例可以提高新实例的创建效率

  \bullet 可以动态增加或减少产品类。

  \bullet 原型模式提高了简化的创建结构。

  \bullet 可以使用深克隆的方式保存对象的状态。

缺点

  \bullet 每一个类都必须配备一个克隆方法,配备克隆方法需要对类的功能进行通盘考虑,这对全新的类来说不是很难,但对已有的类进行改造时,不一定是件容易的事,必须修改其源代码,违背了“开闭原则”。

  \bullet 必须实现 Cloneable 接口。

参考文献:菜鸟教程

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

别偷我的猪_09

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值