四:原型模式详解

原型模式:是指原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象,属于创建型模式

原型模式的核心在于拷贝原型对象。以系统中已存在的一个对象为原型,直接基于内存二进制流进行拷贝,无需再经历耗时的对象初始化过程(不调用构造函数),性能提升许多。当对象的构建过程比较耗时时,可以利用当前系统中已存在的对象作为原型,对其进行克隆(一般是基于二进制流的复制),躲避初始化过程,使得新对象的创建时间大大减少。

原型模式主要适用于以下场景:

1.类初始化消耗资源较多。

2.new产生的一个对象需要非常繁琐的过程(数据准备、访问权限等)

3.构造函数比较复杂。

4.循环体中生产大量对象时。

在 Spring 中,原型模式应用得非常广泛。例如 scope=“prototype”,在我们经常用的JSON.parseObject()也是一种原型模式。

public interface Prototype {
    Prototype clone();
}
    private String name;
    private int age;

    public String getName() {
        return name;
    }

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

    public int getAge() {
        return age;
    }

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

    @Override
    public ConcreatPrototype clone() {
        ConcreatPrototype con = new ConcreatPrototype();
        con.setAge(this.age);
        con.setName(this.name);
        return con;
    }
}
public class Client {
    private Prototype prototype;
    public Client(Prototype prototype){
        this.prototype = prototype;
    }
    public Prototype startClone(Prototype prototype){
        return (Prototype)prototype.clone();
    }
}
public class PrototyprTest {
    public static void main(String[] args) {
        ConcreatPrototype con = new ConcreatPrototype();
        con.setName("rsw");
        con.setAge(18);
        System.out.println(con);
        //创建client对象,这边开始克隆
        Client client = new Client(con);
        ConcreatPrototype prototype = (ConcreatPrototype)client.startClone(con);
        System.out.println(prototype);
        System.out.println("克隆对象中的引用类型的地址"+prototype.getName());
        System.out.println("原对象中的引用类型的地址"+con.getName());
        System.out.println(con.getName() == prototype.getName());
    }
}

从测试的结果name引用地址是相同的,意味着复制的不是值而是引用地址,这就是浅克隆,浅克隆只是完整复制了值的类型数据,没有复制引用的对象,所有的引用对象任然指向原来的对象

还有一种方式,实现Cloneable接口也能实现浅克隆

 

使用实现Serializable接口到达深克隆的目的

public class PrototyprTest {
    public static void main(String[] args) {
        ConcreatPrototype con = new ConcreatPrototype();
        con.setName("rsw");
        con.setAge(18);
        System.out.println(con);
        try{
            ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream();
            ObjectOutputStream objectOutputStream = new ObjectOutputStream(byteArrayOutputStream);
            objectOutputStream.writeObject(con);
            objectOutputStream.close();
            //读流
            ByteArrayInputStream byteArrayInputStream = new ByteArrayInputStream(byteArrayOutputStream.toByteArray());
            ObjectInputStream objectInputStream = new ObjectInputStream(byteArrayInputStream);
            ConcreatPrototype concreatPrototype = (ConcreatPrototype)objectInputStream.readObject();
            objectInputStream.close();
            System.out.println(concreatPrototype);
            System.out.println("---------------------------------");
            System.out.println(con.getName() == concreatPrototype.getName());
        }catch (Exception e){
            e.printStackTrace();
        }
    }
}

如果我们克隆的对象是单例对象,深克隆会破坏单例模式

防止克隆破坏单例的思路禁止克隆便可,要么我们单例类不实现Cloneablie接口,要么我们重写个clone()方法,在该方法中返回单例对象即可

原型模式的优缺点

优点:

1、性能优良,Java自带的 原型模式 是基于内存二进制流的拷贝,比直接new一个对象性能上提升了许多。

2、可以使用深克隆方式保存对象的状态,使用原型模式将对象复制一份并将其状态保存起来,简化了创建对象的过程,以便在需要的时候使用(例如恢复到历史某一状态),可辅助实现撤销操作。

缺点:

1、需要为每一个类配置一个克隆方法。

2、克隆方法位于类的内部,当对已有类进行改造的时候,需要修改代码,违反了开闭原则。

3、在实现深克隆时需要编写较为复杂的代码,而且当对象之间存在多重嵌套引用时,为了实现深克隆,每一层对象对应的类都必须支持深克隆,实现起来会比较麻烦。因此,深拷贝、浅拷贝需要运用得当。

克隆方式:1.序列化 反序列化 2.jsonobject 3浅克隆加赋值

浅克隆:继承Cloneable接口的都是浅克隆。

深克隆两种方式:序列化,转JSON。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值