原型模式和深拷贝、浅拷贝

1. PROS&CONS

PROS

  • 对象拷贝比new实例性能高
  • 简化创建过程

这是因为对象拷贝的过程时,不会调用构造器,通过实现Cloneable接口并调用Object.clone实现对象拷贝。

CONS

  • 必须重写Object.clone
  • 复杂对象进行对象拷贝的风险大,涉及深拷贝,浅拷贝问题

2. 适用场景

  • 类初始化消耗较多资源,循环体生产大量对象时
  • new一个对象过程繁琐,构造函数负债

3. 浅拷贝&深拷贝

浅拷贝

@Setter
@Getter
public class User implements Cloneable {
    private String id;
    private String username;
    private Date birthday;

    @Override
    protected Object clone() throws CloneNotSupportedException {
        return super.clone();
    }

    @Override
    public String toString() {
        return "User{" +
                "id='" + id + '\'' +
                ", username='" + username + '\'' +
                ", birthday=" + birthday +
                '}' + super.toString();
    }
}

user1通过对象拷贝获得user2实例对象,但是user2中的birthday和user1中的birthday指向都了堆中的同一个Date实例对象,即相同引用,属于浅拷贝。这样会导致无论是在user1修改了birthday还是user2修改,birthday都会发生变化。

public static void main(String[] args) throws CloneNotSupportedException {
        Date date1 = new Date(1L);
        User user1 = new User();
        user1.setBirthday(date1);
        User user2 = (User) user1.clone();
        log.info("{}", user1);
        log.info("{}", user2);

        user2.getBirthday().setTime(new Date().getTime());
        log.info("{}", user1);
        log.info("{}", user2);
    }

深拷贝

如何实现深拷贝?通过重现User类的clone方法。

 @Override
    protected Object clone() throws CloneNotSupportedException {
        User user = (User) super.clone();

        // 深克隆
        user.birthday = (Date) user.getBirthday().clone();
        return user;
    }

4. 原型模式破坏单例模式

HungrySingleton hungrySingleton = HungrySingleton.getInstance();
Method method = hungrySingleton.getClass().getDeclaredMethod("clone");
method.setAccessible(true);
HungrySingleton cloneHungrySingleton = (HungrySingleton) method.invoke(hungrySingleton, null);
log.info("{}", hungrySingleton);
log.info("{}", cloneHungrySingleton);

如何解决?

  1. 单例类不实现Cloneable
  2. clone方法返回原单例对象
@Override
protected Object clone() throws CloneNotSupportedException {
    return getInstance();
}

5. 源码案例

  • ArrayList
  • HashMap
  • Mybatis#CacheKey
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 1、什么是深拷贝、什么是浅拷贝 浅拷贝:浅拷贝不会生成新的对象,只会在原对象上增加了一个新的对象引用,两个引用指向...
    勤劳的小手阅读 4,881评论 0 0
  • 设计模式分类 总体来说设计模式分为三大类:创建型模式,共五种:工厂方法模式、抽象工厂模式、单例模式、建造者模式、原...
    lifeline丿毅阅读 4,973评论 0 2
  • 文章部分内容转载自:http://blog.csdn.net/zhangerqing 一、设计模式的分类 总体来说...
    j_cong阅读 6,228评论 0 20
  • 原创文章,转载请标注出处:《Java基础系列-浅拷贝和深拷贝》 一、概述 Java中的拷贝功能是由Object类的...
    唯一浩哥阅读 5,209评论 0 4
  • 窦红梅 焦点初级43期第1044天分享: 有好多人不懂,孩子幼儿受的伤害有可能带一生,有好多家长感觉孩子...
    阿慧_c55c阅读 643评论 0 0