GOF23之原型模式Demo

原型模式是指用原型实例指定创建对象的种类,并通过拷贝这些原型创建新的对象,主要用于对象的复制,它的核心是就是类图中的原型类Prototype。Prototype类需要具备以下两个条件:
实现Cloneable接口、重写Object类中的clone方法。
原型模式是一种比较简单的模式,也非常容易理解,实现一个接口,重写一个方法即完成了原型模式。在实际应用中,原型模式很少单独出现。经常与其他模式混用,他的原型类Prototype也常用抽象类来替代。
以克隆羊为例:
Sheep类

import java.io.Serializable;
import java.util.Date;
public class Sheep implements Cloneable,Serializable {
    private String sname;
    private Date birthday;


    @Override
    protected Object clone() throws CloneNotSupportedException {
        Object obj = super.clone();  //直接调用object对象的clone()方法!
        return obj;
    }

    public String getSname() {
        return sname;
    }

    public void setSname(String sname) {
        this.sname = sname;
    }


    public Date getBirthday() {
        return birthday;
    }

    public void setBirthday(Date birthday) {
        this.birthday = birthday;
    }

    public Sheep(String sname, Date birthday) {
        super();
        this.sname = sname;
        this.birthday = birthday;
    }

    public Sheep() {}
}

Client测试类:

public class Client {
    public static void main(String[] args) throws Exception {
        Date date = new Date(12312321331L);
        Sheep s1 = new Sheep("我是一只羊",date);
        System.out.println(s1);
        System.out.println(s1.getSname());
        System.out.println(s1.getBirthday());

        Sheep s2 = (Sheep) s1.clone();
        s2.setSname("我是另一只羊");

        date.setTime(23432432423L);

        System.out.println(s1.getBirthday());
        System.out.println(s2);
        System.out.println(s2.getSname());
        System.out.println(s2.getBirthday());

    }
}

运行结果:
com.gujian.prototype.Sheep@6e1408
我是一只羊
Sat May 23 20:05:21 CST 1970
Tue Sep 29 13:00:32 CST 1970
com.gujian.prototype.Sheep@21b6d
我是另一只羊
Tue Sep 29 13:00:32 CST 1970
可以看出,s1的birthday属性变化会导致s2的也变化,这是一种浅复制

备注:深拷贝与浅拷贝问题中,会发生深拷贝的有java中的8中基本类型以及他们的封装类型,另外还有String类型。其余的都是浅拷贝。

如果要进行深复制,需要将属性也进行克隆。改造如下:

import java.util.Date;

//测试深复制
public class Sheep2 implements Cloneable {   
    private String sname;
    private Date birthday;

    @Override
    protected Object clone() throws CloneNotSupportedException {
        Object obj = super.clone();  //直接调用object对象的clone()方法!

        //添加如下代码实现深复制(deep Clone)
        Sheep2 s = (Sheep2) obj;
        s.birthday = (Date) this.birthday.clone();  //把属性也进行克隆!

        return obj;
    }


    public String getSname() {
        return sname;
    }


    public void setSname(String sname) {
        this.sname = sname;
    }


    public Date getBirthday() {
        return birthday;
    }

    public void setBirthday(Date birthday) {
        this.birthday = birthday;
    }

    public Sheep2(String sname, Date birthday) {
        super();
        this.sname = sname;
        this.birthday = birthday;
    }

    public Sheep2() {}
}

Client2 测试类:

public class Client2 {
    public static void main(String[] args) throws CloneNotSupportedException {
        Date date = new Date(12312321331L);
        Sheep2 s1 = new Sheep2("我是一只羊",date);
        Sheep2 s2 = (Sheep2) s1.clone();   //实现深复制。s2对象的birthday是一个新对象!

        System.out.println(s1);
        System.out.println(s1.getSname());
        System.out.println(s1.getBirthday());

        date.setTime(23432432423L);

        System.out.println(s1.getBirthday());

        s2.setSname("我是另一只养");
        System.out.println(s2);
        System.out.println(s2.getSname());
        System.out.println(s2.getBirthday());
    }
}

运行结果:
com.gujian.prototype.Sheep2@e53108
我是一只羊
Sat May 23 20:05:21 CST 1970
Tue Sep 29 13:00:32 CST 1970
com.gujian.prototype.Sheep2@21b6d
我是另一只养
Sat May 23 20:05:21 CST 1970
可以看出s1对象的修改并没有影响到s2。

还可以使用使用序列化和反序列化的方式实现深复制。
测试代码:

public class Client3 {
    public static void main(String[] args) throws CloneNotSupportedException, Exception {
        Date date = new Date(12312321331L);
        Sheep s1 = new Sheep("我是一只羊",date);
        System.out.println(s1);
        System.out.println(s1.getSname());
        System.out.println(s1.getBirthday());

//      使用序列化和反序列化实现深复制
        ByteArrayOutputStream bos = new ByteArrayOutputStream();
        ObjectOutputStream    oos = new ObjectOutputStream(bos);
        oos.writeObject(s1);
        byte[] bytes = bos.toByteArray();

        ByteArrayInputStream  bis = new ByteArrayInputStream(bytes);
        ObjectInputStream     ois = new ObjectInputStream(bis);

        Sheep s2 = (Sheep) ois.readObject();   //克隆好的对象!

        System.out.println("修改原型对象的属性值");  
        date.setTime(23432432423L);

        System.out.println(s1.getBirthday());

        s2.setSname("我是另一只羊");
        System.out.println(s2);
        System.out.println(s2.getSname());
        System.out.println(s2.getBirthday());
    }
}

运行结果:
com.gujian.prototype.Sheep@19189e1
我是一只羊
Sat May 23 20:05:21 CST 1970
修改原型对象的属性值
Tue Sep 29 13:00:32 CST 1970
com.gujian.prototype.Sheep@f73c1
我是另一只羊
Sat May 23 20:05:21 CST 1970
同样s1对象的修改并没有影响到s2。

原型模式的使用场景:
先比较普通new方式创建对象和clone方式创建对象的效率差异。
测试代码:

public class Client4 {

    public static void testNew(int size){
        long start = System.currentTimeMillis();
        for(int i=0;i<size;i++){
            Laptop t = new Laptop();
        }
        long end = System.currentTimeMillis();
        System.out.println("new的方式创建耗时:"+(end-start));
    }

    public static void testClone(int size) throws CloneNotSupportedException{
        long start = System.currentTimeMillis();
        Laptop t = new Laptop();
        for(int i=0;i<size;i++){
            Laptop temp = (Laptop) t.clone();
        }
        long end = System.currentTimeMillis();
        System.out.println("clone的方式创建耗时:"+(end-start));
    }


    public static void main(String[] args) throws Exception {   
        testNew(1000);
        testClone(1000);
    }
}


class Laptop implements Cloneable {  //笔记本电脑
    public Laptop() {
        try {
            Thread.sleep(10);  //模拟创建对象耗时的过程!
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }

    @Override
    protected Object clone() throws CloneNotSupportedException {
        Object obj = super.clone();  //直接调用object对象的clone()方法!
        return obj;
    }

}

运行结果:
new的方式创建耗时:10328
clone的方式创建耗时:10
如果需要短时间创建大量对象,并且new的过程比较耗时。则可以考虑使用原型模式。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值