Java中克隆万物的Clone方法!

50 篇文章 0 订阅
44 篇文章 0 订阅
本文深入探讨了Java中Object类的clone方法,讲解了如何使用以及为何需要实现Cloneable接口。通过实例展示了浅拷贝的工作原理,解释了为什么在某些情况下对象属性为对象时,clone后的对象仍然可能存在共享状态。同时,文章还介绍了如何实现深拷贝以确保完全独立的副本。
摘要由CSDN通过智能技术生成

在这里插入图片描述

Hello,大家好,我是洪爵呀!今天想和大家聊一下Java的Object类中的Clone方法,它有什么用,我自己new一个不行吗?

​ 不知道大家有没有用过这个方法,洪爵在面试的时候,经常被问到Object中有哪些方法?常常答不出来呀,因此苦恼,所以决定好好专研下Object里的方法,一个个抓出来!

​ 我们先来看一下clone源码:

public class Object {
    protected native Object clone() throws CloneNotSupportedException;
}

Object类中有一个protected修饰的native方法clone,并且是throws CloneNotSupportedException的。protected修饰意味着继承Object的类都可以使用这个方法,基本所有类都会继承自Object类,这个没什么好说的。native方法说明是使用非java语言编写的方法,如果我们要使用clone方法,必须要实现Cloneable接口,如果不重写的话,会抛出CloneNotSupportedException的异常。

public interface Cloneable {
}

​ 上面是Cloneable的源码,可以发现它是一个空的接口,但是如果大家点进去查看注释,会发现注释中已经告诉你必须要实现这个接口才能去使用clone方法,实现这个接口相当于一个标志,旨在告诉jvm,是可以clone的了,不要再给我抛出CloneNotSupportedException啦。

​ 那为什么要使用clone呢?

​ clone的想法是创建一个副本,在最坏的情况下需要使用两倍的内存。clone方法是浅拷贝的,也就是说对象中属性是对象的只会拷贝它的引用地址,而不会重新为这个对象分配内存,像基本数据类型int,float,double这些就会直接复制值。

​ 所以说如果说这个对象中的属性都是对象,那么很有可能这个被clone出来的对象不需要占用太多的内存。

​ 这样说可能比较抽象,我们来看一个生动的例子。

public class Main {
    public static void main(String[] args) throws CloneNotSupportedException {
        TestClone testClone = new TestClone();
        testClone.setInteger(new Integer(128));
        testClone.setInts(2);
        testClone.setStr("3");
        testClone.setTestClone(new TestClone());
        testClone.getTestClone().setStr("123");
        testClone.getTestClone().setInteger(1);
        testClone.getTestClone().setInts(33);
        TestClone testClone1 = (TestClone) testClone.clone();
        System.out.println(testClone.toString());
        System.out.println(testClone1.toString());
        testClone.setStr("test1");
        testClone.setInts(333);
        testClone.setInteger(4444);
        testClone.getTestClone().setStr("123444");
        testClone.getTestClone().setInteger(111111);
        testClone.getTestClone().setInts(33333333);
        System.out.println(testClone.toString());
        System.out.println(testClone1.toString());

    }

}
/*
TestClone{str='3', ints=2, integer=128, testClone=TestClone{str='123', ints=33, integer=1, testClone=null}}
TestClone{str='3', ints=2, integer=128, testClone=TestClone{str='123', ints=33, integer=1, testClone=null}}
TestClone{str='test1', ints=333, integer=4444, testClone=TestClone{str='123444', ints=33333333, integer=111111, testClone=null}}
TestClone{str='3', ints=2, integer=128, testClone=TestClone{str='123444', ints=33333333, integer=111111, testClone=null}}
*/
class TestClone implements Cloneable {
    private String str;
    private int ints;
    private Integer integer;
    private TestClone testClone;

    public String getStr() {
        return str;
    }

    public void setStr(String str) {
        this.str = str;
    }

    public int getInts() {
        return ints;
    }

    public void setInts(int ints) {
        this.ints = ints;
    }

    public Integer getInteger() {
        return integer;
    }

    public void setInteger(Integer integer) {
        this.integer = integer;
    }

    public TestClone getTestClone() {
        return testClone;
    }

    public void setTestClone(TestClone testClone) {
        this.testClone = testClone;
    }


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

    @Override
    public String toString() {
        return "TestClone{" +
                "str='" + str + '\'' +
                ", ints=" + ints +
                ", integer=" + integer +
                ", testClone=" + testClone +
                '}';
    }
}

可以发现String和Integer都是对象呀,为啥修改之后,testclone1打印出来的和testclone不一样呀?因为String和Integer,还有Long等这些类,都是通过final去修饰value的,一旦修改都是直接去new一个对象出来,所以自然testclone和testclone1打印出来的不是同一个对象啦。

​ 当然了,这只是浅拷贝,还有深拷贝,深拷贝则是需要自己去实现clone方法了,我们拿之前的例子来说明,如果上面TestClone对象要实现深拷贝要如何实现clone方法。

	@Override
    protected Object clone() throws CloneNotSupportedException {
        TestClone testClone = null;
        try {
            testClone = (TestClone) super.clone();
            
            TestClone testClone1 = testClone.getTestClone();
            TestClone testClone2 = new TestClone();
            
            testClone2.setInts(testClone1.getInts());
            testClone2.setInteger(testClone1.getInteger());
            testClone2.setStr(testClone1.getStr());
            testClone2.setTestClone(testClone1.getTestClone());
            
            testClone.setTestClone(testClone2);
        } catch (CloneNotSupportedException e) {
            e.printStackTrace();
        }
        return testClone;
    }

​ 这个时候如果你再去运行上面那套代码,你就会发现不一样的地方啦!赶紧去尝试一波吧~

​ OK,本章clone就讲到这里啦,希望对你有所帮助!

​愿每个人都能带着怀疑的态度去阅读文章并探究其中原理。

道阻且长,往事作序,来日为章。

期待我们下一次相遇!
在这里插入图片描述

评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

KnightHONG

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值