java深入理解浅拷贝和深拷贝

简介: java深入理解浅拷贝和深拷贝

目录



简介


拷贝对象是java中经常会遇到的问题。java中存在两种类型,基础类型和引用类型。


java的赋值都是传值的,对于基础类型来说,会拷贝具体的内容,但是对于引用对象来说,存储的这个值只是指向实际对象的地址,拷贝也只会拷贝引用地址。


因为引用对象的存在,所以经常会出现和预期不一样的情况。


本文将会深入的探讨一下在拷贝对象中会出现的浅拷贝和深拷贝的情况。


拷贝接口


java中所有的对象都是继承自java.lang.Object。Object对象中提供了一个clone方法,来供我们对java对象进行拷贝。


protected native Object clone() throws CloneNotSupportedException;


这个clone方法是native的,所以不需要我们来实现,但是注意clone方法还是protected,这意味着clone方法只能在java.lang包或者其子类可见。


如果我们想要在一个程序中调用某个对象的clone方法则是不可以的。因为clone方法是定义在Object中的,该对象并没有对外可见的clone方法。


JDK的建议是让我们去实现接口Cloneable,实现了这个接口就表示这个对象可以调用Object的clone方法。


注意,即使你实现了Cloneable接口,还是无法在外部程序中调用该对象的clone方法:


public interface Cloneable {
}


因为Cloneable是空的,明没有强制要你去实现clone方法。


这是JDK在设计上的问题,导致clone方法并不像预期那么好用。


首先clone只是对象的拷贝,它只是简单的拷贝对象,而不会去执行对象的构造函数。


其次clone会导致浅拷贝的问题。


使用clone导致的浅拷贝


我们举个clone产生的浅拷贝的例子,我们定义一个对象中的对象,然后尝试拷贝:


@Data
public class Address implements Cloneable{
    private String name;
    //不是好的方式
    @Override
    protected Object clone() throws CloneNotSupportedException {
        return super.clone();
    }
}


@Data
public class CustUser implements Cloneable{
    private String firstName;
    private String lastName;
    private Address address;
    private String[] cars;
    @Override
    public Object clone() throws CloneNotSupportedException{
            return super.clone();
    }
}


上面的例子中,我们定义了CustUser和Address。


public void testShallowCopy() throws CloneNotSupportedException {
        Address address= new Address();
        address.setName("北京天安门");
        CustUser custUser = new CustUser();
        custUser.setAddress(address);
        custUser.setLastName("李");
        custUser.setFirstName("雷");
        String[] cars = new String[]{"别克","路虎"};
        custUser.setCars(cars);
        CustUser custUserCopy=(CustUser) custUser.clone();
        custUserCopy.setFirstName("梅梅");
        custUserCopy.setLastName("韩");
        custUserCopy.getAddress().setName("北京颐和园");
        custUserCopy.getCars()[0]="奥迪";
        log.info("{}",custUser);
        log.info("{}",custUserCopy);
    }


浅拷贝我们只调用了CustUser的clone方法。看下输出结果:


CustUser(firstName=雷, lastName=李, address=Address(name=北京颐和园), cars=[奥迪, 路虎])
CustUser(firstName=梅梅, lastName=韩, address=Address(name=北京颐和园), cars=[奥迪, 路虎])


我们可以看到拷贝之后的Address变化会影响到被拷贝的对象。


上面的例子我们还要关注两个点:第一点String是不可变的。不管是拷贝还是赋值,String都是不可变的。


第二点,上面的例子中我们定义了一个数组,可以看到如果只是调用clone的话,数组也是浅拷贝。


使用clone的深拷贝


要使用深拷贝,只需要修改CustUser的构造函数就可以了:


//不是很好的使用方式
    @Override
    public Object clone() throws CloneNotSupportedException{
        CustUserDeep custUserDeep=(CustUserDeep)super.clone();
        custUserDeep.address=(Address)address.clone();
        custUserDeep.cars=cars.clone();
            return custUserDeep;
    }


在重写的clone方法中,我们分别调用了CustUser,Address和数组的clone方法来进行拷贝。


再运行一次上面的测试代码:


CustUserDeep(firstName=雷, lastName=李, address=Address(name=北京天安门), cars=[别克, 路虎])
CustUserDeep(firstName=梅梅, lastName=韩, address=Address(name=北京颐和园), cars=[奥迪, 路虎])


可以看到address和cars是不同的,这表示我们的深拷贝是成功的。


不要overridden clone


上面的例子我们是通过overridden Object的clone方法来实现的。


但是最佳实践是不要overridden clone。那我们怎么做呢?


使用构造函数来构建新的对象:


//好的方式
    Address(Address address){
        this.name=address.name;
    }


//很好的方式
    CustUserDeep(CustUserDeep custUserDeep){
    this.firstName=custUserDeep.firstName;
    this.lastName=custUserDeep.lastName;
    this.cars=custUserDeep.getCars().clone();
    this.address=new Address(custUserDeep.getAddress());
    }


据说数组直接用clone来拷贝会更快,也可以使用下面的方式来拷贝数组:


this.cars= Arrays.copyOf(custUserDeep.getCars(),custUserDeep.getCars().length);


总结


本文讲解了浅拷贝和深拷贝的应用,并对clone方法做了深入的探讨。


本文的例子https://github.com/ddean2009/learn-java-base-9-to-20

相关文章
|
6月前
|
Java
Java的浅拷贝与深拷贝
Java的浅拷贝与深拷贝
53 0
|
6月前
|
Java Apache
Java中的深拷贝与浅拷贝
Java中的深拷贝与浅拷贝
44 0
|
5月前
|
Java
“深入探讨Java中的对象拷贝:浅拷贝与深拷贝的差异与应用“
“深入探讨Java中的对象拷贝:浅拷贝与深拷贝的差异与应用“
|
5月前
|
存储 Dubbo Java
一篇文章讲明白Java的深拷贝和浅拷贝
一篇文章讲明白Java的深拷贝和浅拷贝
50 0
|
6月前
|
Java
【JAVA面试题】什么是深拷贝?什么是浅拷贝?
【JAVA面试题】什么是深拷贝?什么是浅拷贝?
|
Java
Java深拷贝和浅拷贝Map对象
Java深拷贝和浅拷贝Map对象
190 0
|
Java
【java面试题】- java深拷贝和浅拷贝区别?什么是引用拷贝?
java深拷贝和浅拷贝区别?什么是引用拷贝?
86 0
|
6天前
|
安全 Java
java 中 i++ 到底是否线程安全?
本文通过实例探讨了 `i++` 在多线程环境下的线程安全性问题。首先,使用 100 个线程分别执行 10000 次 `i++` 操作,发现最终结果小于预期的 1000000,证明 `i++` 是线程不安全的。接着,介绍了两种解决方法:使用 `synchronized` 关键字加锁和使用 `AtomicInteger` 类。其中,`AtomicInteger` 通过 `CAS` 操作实现了高效的线程安全。最后,通过分析字节码和源码,解释了 `i++` 为何线程不安全以及 `AtomicInteger` 如何保证线程安全。
java 中 i++ 到底是否线程安全?
|
1天前
|
存储 设计模式 分布式计算
Java中的多线程编程:并发与并行的深度解析####
在当今软件开发领域,多线程编程已成为提升应用性能、响应速度及资源利用率的关键手段之一。本文将深入探讨Java平台上的多线程机制,从基础概念到高级应用,全面解析并发与并行编程的核心理念、实现方式及其在实际项目中的应用策略。不同于常规摘要的简洁概述,本文旨在通过详尽的技术剖析,为读者构建一个系统化的多线程知识框架,辅以生动实例,让抽象概念具体化,复杂问题简单化。 ####