【设计模式】四、原型模式(5分钟轻松搞定)

前言

原型模式核心在于实例的拷贝。以系统内已存在的实例,直接基于二进制流进行拷贝,无需经过耗时的对象初始化过程。不调用构造函数。
实现方法十分简单,是需要实现Cloneable接口,并覆盖clone方法即可。

public class PersonPrototype implements Cloneable{

    private Integer age;

    private String 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;
    }

    @Override
    protected PersonPrototype clone() {
        try {
            return (PersonPrototype) super.clone();
        } catch (CloneNotSupportedException e) {
            e.printStackTrace();
        }
        return null;
    }

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

测试类

public static void main(String[] args) {
        PersonPrototype p1 = new PersonPrototype();
        p1.setAge(11);
        p1.setName("a");
        PersonPrototype p2 = p1.clone();
        System.out.println(p1);
        System.out.println(p2);
 }

输出结果:
在这里插入图片描述
接下来,我们添加一个爱好属性hobbies

public class PersonPrototype implements Cloneable{

    private Integer age;

    private String name;

    private List<String> hobbies;
    .....(省略其余代码)
 }

测试代码

PersonPrototype p1 = new PersonPrototype();
        p1.setAge(11);
        p1.setName("a");
        List<String> list = new ArrayList<>();
        list.add("画画");
        list.add("书法");
        p1.setHobbies(list);
        PersonPrototype p2 = p1.clone();
        p2.setName("b");
        System.out.println(p1);
        System.out.println(p2);
        p1.getHobbies().add("钢琴");
        System.out.println(p1);
        System.out.println(p2);

运行结果:
在这里插入图片描述
我们给复制后的克隆对象新增一项爱好,发现原型对象也被修改了。这显然不符合我们的预期。以为我们希望克隆出来的对象应该和原型对象是两个相互独立的对象,不应该再有任何联系。从结果我们分析,hobbies共用了相同的内存地址。这意味着复制的不是值,而是引用地址。这就是我们常说的浅克隆

浅克隆

浅克隆只是完整的复制值类型的数据,没有赋值引用对象。也就是说,所有引用对象仍只想原来的对象。这显然不是我们想要的。下面我们来探讨如果进行深度克隆。

深克隆

在原来的基础上,增加deepClone方法,同时实现Serializable接口。

public class PersonDeepPrototype implements Cloneable, Serializable {
	....
	    public PersonDeepPrototype deepClone(){
        try {
            ByteArrayOutputStream bos = new ByteArrayOutputStream();
            ObjectOutputStream oos = new ObjectOutputStream(bos);
            oos.writeObject(this);
            ByteArrayInputStream bis = new ByteArrayInputStream(bos.toByteArray());
            ObjectInputStream ois = new ObjectInputStream(bis);
            return (PersonDeepPrototype) ois.readObject();
        }catch (Exception e){
            e.printStackTrace();
        }
        return null;
    }
	....
}

测试代码:

		PersonDeepPrototype  p1 = new PersonDeepPrototype();
        p1.setAge(11);
        p1.setName("a");
        List<String> list = new ArrayList<>();
        list.add("画画");
        list.add("书法");
        p1.setHobbies(list);
        PersonDeepPrototype p2 = p1.deepClone();
        p2.setName("b");
        System.out.println(p1);
        System.out.println(p2);
        p1.getHobbies().add("钢琴");
        System.out.println(p1);
        System.out.println(p2);

运行结果:
在这里插入图片描述
通过序列化,这次克隆的对象已经完全独立出来。得到了我们预期的结果。

克隆破坏单例

为了防止我们的单例被克隆,简单的有两种方式。

  1. 对象不实现Cloneable 接口
  2. 重写clone方法,方法中直接返回当前实例。

原型模式的优缺点

优点

  • 性能优良
  • 简化对象创建流程

缺点

  • 为每个类配置一个克隆方法
  • 对已有的类改造需要修改代码
  • 深度克隆编写起来比较复杂

源码地址

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值