原型模式

简介

原型模式属于创建型模式的一种,它是指通过拷贝自己去创建一些新的对象。拷贝分为深拷贝和浅拷贝,但核心都在于重写clone方法,原型模式相比较于传统的一直new的方式复制效率上有所提升,并且有更好的扩展性。

实现

浅拷贝:原型中如果有引用类型,拷贝的对象和原型共享该引用类型,即在拷贝的时候在内存中并没有新创建一个引用类型对象

深拷贝:原型和拷贝的对象不共享引用类型

import java.io.*;

/**
 * 原型模式
 * 用来解决克隆对象的问题
 */
public class ProtoType implements Cloneable {

    String name;
    int age;
    Proto proto;

    public ProtoType(String name, int age, Proto proto) {
        this.name = name;
        this.age = age;
        this.proto = proto;
    }

    @Override
    public String toString() {
        return "ProtoType{" +
                "name='" + name + '\'' +
                ", age=" + age +
                ", proto=" + proto +
                '}';
    }

    //浅拷贝:属性如果是引用类型,则拷贝出来的对象和原有的对象共享同一个引用类型
    @Override
    public ProtoType clone() throws CloneNotSupportedException {
        ProtoType protoType = (ProtoType) super.clone();
        return protoType;
    }

}


class DeepProtoType implements Serializable, Cloneable {

    String name;
    int age;
    Proto proto;

    public DeepProtoType(String name, int age, Proto proto) {
        this.name = name;
        this.age = age;
        this.proto = proto;
    }

    //深拷贝的两种方式:
    //1.重写clone方法:这种方法的缺点是如果引用类型太多的话就要写很多,不推荐
//    @Override
//    public Object clone() throws CloneNotSupportedException {
//        DeepProtoType protoType = (DeepProtoType) super.clone();
//        protoType.proto = this.proto.clone();
//        return protoType;
//    }

    //2.通过对象序列化实现
    @Override
    public Object clone() throws CloneNotSupportedException {

        //创建流对象
        ByteArrayInputStream bis = null;
        ByteArrayOutputStream bos = null;
        ObjectInputStream ois = null;
        ObjectOutputStream oos = null;


        try{
            //序列化
            bos = new ByteArrayOutputStream();
            oos = new ObjectOutputStream(bos);
            oos.writeObject(this);

            //反序列化
            bis = new ByteArrayInputStream(bos.toByteArray());
            ois = new ObjectInputStream(bis);
            DeepProtoType deepProtoType =  (DeepProtoType) ois.readObject();

            return deepProtoType;

        }catch(Exception e){
            e.printStackTrace();
        }finally {
            try {
                bis.close();
                bos.close();
                ois.close();
                oos.close();
            } catch (IOException e) {
                e.printStackTrace();
            }
        }

        return null;
    }

    @Override
    public String toString() {
        return "DeepProtoType{" +
                "name='" + name + '\'' +
                ", age=" + age +
                ", proto=" + proto +
                '}';
    }


}

class Proto implements Serializable,Cloneable {

    String name;

    public Proto(String name) {
        this.name = name;
    }

    @Override
    public Proto clone() throws CloneNotSupportedException {
        return (Proto) super.clone();
    }

    @Override
    public String toString() {
        return "Proto{" +
                "name='" + name + '\'' +
                '}';
    }
}

class test {

    public static void main(String[] args) throws Exception {

        ProtoType protoType1 = new ProtoType("测试", 1, new Proto("附属"));
        ProtoType protoType2 = (ProtoType) protoType1.clone();

        System.out.println(protoType1 == protoType2);//false
        System.out.println(protoType1.proto == protoType2.proto);//true

        DeepProtoType deepProtoType1 = new DeepProtoType("测试",1,new Proto("附属"));
        DeepProtoType deepProtoType2 = (DeepProtoType) deepProtoType1.clone();

        System.out.println(deepProtoType1 == deepProtoType2);//false
        System.out.println(deepProtoType1.proto == deepProtoType2.proto);//false


    }
}

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值