【Java 设计模式 · 创建型】浅拷贝 & 深拷贝、原型模式(Prototype Pattern)

创建型模式:关注对象的创建过程,它描述如何将对象的创建、使用分离,让用户无需关心对象的创建细节,从而降低系统的耦合度,让设计方案易于修改、扩展。

一、概述

原型模式(Prototype Pattern):使用原型实例指定待创建对象的类型,并且通过复制这个原型来创建新的对象。

原型模式是一种特殊的创建型模式,它通过复制一个已有对象来获取更多的相同 / 相似的对象。

二、结构

  • Prototype(抽象原型类):
    声明克隆方法的接口,所有具体原型类的公共父类,可以是接口、抽象类、具体类。
  • ConcretePrototype(具体原型类):
    实现原型中的克隆方法,返回自己的一个克隆对象。

原型模式

三、拷贝问题

1、浅拷贝

在浅拷贝中,原型对象的成员变量有两种情况:

  • 值类型:复制一份给克隆对象
  • 引用类型:复制引用给克隆对象(不同引用指向同一类型同一对象)

2、深拷贝

在深拷贝中,原型对象的成员变量有两种情况:

  • 值类型:复制一份给克隆对象
  • 引用类型:复制引用对象给克隆对象(不同引用指向同一类型不同对象)

四、举例实现

定义原型对象Person,内部有属性 Car 对象。

1、浅拷贝

实例代码:

public class Car {
    private Integer id;
    private String brand;
    private String color;
    private Double price;

    /*
     * 省略所有属性的 Setter、Getter 方法,重写 toString() 方法,无参、全参 构造器
     */

}
public class Person implements Cloneable {
    private Integer id;
    private String name;
    private Car car;

    /*
     * 省略所有属性的 Setter、Getter 方法,重写 toString() 方法,无参、全参 构造器
     */

    @Override
    public Person clone() throws CloneNotSupportedException {
        Object obj = null;
        try {
            obj = super.clone();
            return (Person) obj;
        } catch (CloneNotSupportedException e) {
            System.out.println("不支持拷贝!");
            return null;
        }
    }
}

测试代码:

Car car = new Car(1,"Benz","红色",200000.00);
Person p = new Person(1,"张三",car);
Person p2 = p.clone();
System.out.println("是否为同一引用:" + (p == p2));
System.out.println("是否为同一种类型:" + p.equals(p2));

p.getCar().setColor("蓝色");

System.out.println(p);
System.out.println(p2);

浅拷贝

2、深拷贝

实例代码:

public class Car implements Serializable {
    private Integer id;
    private String brand;
    private String color;
    private Double price;

    /*
     * 省略所有属性的 Setter、Getter 方法,重写 toString() 方法,无参、全参 构造器
     */
     
}
public class Person implements Serializable {
    private Integer id;
    private String name;
    private Car car;

    /*
     * 省略所有属性的 Setter、Getter 方法,重写 toString() 方法,无参、全参 构造器
     */

    public Person deepClone() throws IOException, ClassNotFoundException {
        //将对象写入流
        ByteArrayOutputStream bao = new ByteArrayOutputStream();
        ObjectOutputStream oos = new ObjectOutputStream(bao);
        oos.writeObject(this);
        //将对象从流中取出
        ByteArrayInputStream bis = new ByteArrayInputStream(bao.toByteArray());
        ObjectInputStream ois = new ObjectInputStream(bis);
        return (Person) ois.readObject();
    }
}

测试代码:

Car car = new Car(1,"Benz","红色",200000.00);
Person p = new Person(1,"张三",car);
Person p2 = p.deepClone();
System.out.println("是否为同一引用:" + (p == p2));
System.out.println("是否为同一种类型:" + p.equals(p2));

p.getCar().setColor("蓝色");

System.out.println(p);
System.out.println(p2);

深拷贝

五、原型管理器

原型管理器(Prototype Manager) 将多个原型对象存储在一个集合中,以供拷贝,便于扩展。

六、特点

☯ 优点

  • 提高已有实例对象的创建效率
  • 扩展性较好,可以将具体原型写入配置文件
  • 提供了简化的创建结构,拥有原型可以无需额外的工厂类
  • 可以使用深拷贝保存对象的状态,支持将对象恢复到某一历史状态

☯ 缺点

  • 需要为每个类配备一个克隆方法,违背了开闭原则
  • 在深拷贝须编写较为复杂的代码
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值