【设计模式】原型模式(Prototype Pattern) clone和序列化的浅拷贝和深拷贝

目录

 

一.介绍

二.场景

三.实现

四.dome代码


一.介绍

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

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

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

主要解决:在运行期建立和删除原型。

何时使用: 1、当一个系统应该独立于它的产品创建,构成和表示时。 2、当要实例化的类是在运行时刻指定时,例如,通过动态装载。 3、为了避免创建一个与产品类层次平行的工厂类层次时。 4、当一个类的实例只能有几个不同状态组合中的一种时。建立相应数目的原型并克隆它们可能比每次用合适的状态手工实例化该类更方便一些。

如何解决:利用已有的一个原型对象,快速地生成和原型对象一样的实例。

关键代码: 1、实现克隆操作,在 JAVA 继承 Cloneable,重写 clone(),在 .NET 中可以使用 Object 类的 MemberwiseClone() 方法来实现对象的浅拷贝或通过序列化的方式来实现深拷贝。 2、原型模式同样用于隔离类对象的使用者和具体类型(易变类)之间的耦合关系,它同样要求这些"易变类"拥有稳定的接口。

应用实例: 1、细胞分裂。 2、JAVA 中的 Object clone() 方法。

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

缺点: 1、配备克隆方法需要对类的功能进行通盘考虑,这对于全新的类不是很难,但对于已有的类不一定很容易,特别当一个类引用不支持串行化的间接对象,或者引用含有循环结构的时候。 2、必须实现 Cloneable 接口。

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

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

二.场景

公司组织架构

三.实现

1.儿子类

package xyz.hashdog.prototype;


import java.io.Serializable;

public class Son implements Serializable {
    private Integer age;

    private String name;

    public Son(Integer age, String name) {
        this.age = age;
        this.name = name;
    }

    public Integer getAge() {
        return age;
    }

    public void setAge(Integer age) {
        this.age = age;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }
}

2.父亲类

package xyz.hashdog.prototype;

import java.io.*;
import java.util.ArrayList;
import java.util.List;

/**
 * clone方法拷贝需要实现Cloneable接口,这里用接口传递
 * 序列化拷贝需要实现Serializable
 */
public class Father implements HashDogCloneable<Father> ,Serializable{

    private List<Son> sons;

    public List<Son> getSons() {
        return sons;
    }

    public void setSons(List<Son> sons) {
        this.sons = sons;
    }

    /**
     * clone实现浅拷贝
     * clone重写方法也能深拷贝,但是很容易造成浅拷贝
     * 以下逻辑,List中的son元素就不是深拷贝
     * @return
     * @throws CloneNotSupportedException
     */
    @Override
    protected Object clone() throws CloneNotSupportedException {
        Father clone = (Father)super.clone();
        List cloneSons = (List)((ArrayList) sons).clone();
        clone.sons=cloneSons;
        return clone;
    }


    /**
     * 序列化深拷贝
     * @return
     * @throws IOException
     * @throws ClassNotFoundException
     */
    @Override
    public Father copy() throws IOException, ClassNotFoundException {
        //通过序列化方法实现深拷贝
        ByteArrayOutputStream bos=new ByteArrayOutputStream();
        ObjectOutputStream oos=new ObjectOutputStream(bos);
        oos.writeObject(this);
        oos.flush();
        ObjectInputStream ois=new ObjectInputStream(new ByteArrayInputStream(bos.toByteArray()));
        Father f=(Father)ois.readObject();
        return f;
    }
}

3.clone接口扩展

package xyz.hashdog.prototype;

import java.io.IOException;

public interface HashDogCloneable<T> extends Cloneable {

    T copy() throws IOException, ClassNotFoundException;
}

4.场景类

package xyz.hashdog.prototype;

import java.io.IOException;
import java.util.ArrayList;
import java.util.List;

/**
 * 原型模式
 * 可以发现clone方法拷贝的father对象和father的list<son>属性是深拷贝
 * 但是list中的son元素是浅拷贝
 * 而使用序列化都是深拷贝
 * 如果用clone方式实现,深拷贝,每次添加元素都得考虑该元素的clone方法重写
 */
public class Main {
    public static void main(String[] args) throws CloneNotSupportedException, IOException, ClassNotFoundException {

        Father f = new Father();
        List<Son> sons = new ArrayList<>();
        sons.add(new Son(10,"大儿子"));
        sons.add(new Son(5,"小儿子"));
        f.setSons(sons);

        System.out.println("***************clone浅拷贝*****************");
        Father f2 = (Father)f.clone();
        System.out.println(f==f2);
        System.out.println(f.getSons()==f2.getSons());
        System.out.println(f.getSons().get(0)==f2.getSons().get(0));

        System.out.println("***************序列化深拷贝*****************");
        Father copy = f.copy();
        System.out.println(f==copy);
        System.out.println(f.getSons()==copy.getSons());
        System.out.println(f.getSons().get(0)==copy.getSons().get(0));


    }
}

 

5.执行

 

四.dome代码

https://download.csdn.net/download/corleone_4ever/12674674

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值