Object中clone,toString和getClass方法的分析(画图深度剖析浅克隆,深克隆)

public String toString() {
    return getClass().getName() + "@" + Integer.toHexString(hashCode());
}

getClass()获取类对象
getName()以String形式返回类对象的名称
Integer.toHexString(hashCode())以对象的哈希码为参数,以16进制无符号整数形式返回此哈希码的字符串表示形式;
在实体类中我们一般会重写toString方法。
(1、在大量代码调试的时候它的作用就凸显了,可以很方便的查看这个实体类含带的属性的值,无需一层层翻看实体类的属性找值了。有些复杂的实体变量很难在调试的时候找到它含带的值,这时候toString()就显得很有用,容易监控实体类属性值变动和走向。
2、在输出显示实体类的易查看。实体类可以像String一样方便的输出显示。
在这里插入图片描述
clone方法的分析

为什么要克隆对象???直接new出来一个不好吗???
克隆的对象可能包含一些已经修改过的属性,而new出来的对象的属性都还是初始化时候的值,所以当需要一个新的对象来保存当前对象的“状态”就靠clone方法了。那么我把这个对象的临时属性一个一个的赋值给我新new的对象不也行嘛?可以是可以,但是一来麻烦不说,二来,大家通过上面的源码都发现了clone是一个native方法,就是快啊,在底层实现的。
clone方法克隆的对象跟原来的对象同时独立存在的。
先介绍一下两种不同的克隆方法,浅克隆(ShallowClone)和深克隆(DeepClone)。
浅克隆:

  1. 被复制的类需要实现Clonenable接口(不实现的话在调用clone方法会抛出CloneNotSupportedException异常)。

  2. 该接口为标记接口,接口里只有一句话,不含任何方法。在这里插入图片描述

  3. 覆盖clone()方法,访问修饰符设为public。方法中调用super.clone()方法得到需要的复制对象。

public class Student  implements  Cloneable{
    private int number;

    public int getNumber() {
        return number;
    }

    public void setNumber(int number) {
        this.number = number;
    }

   @Override
   public Object clone() {
       Student stu = null;//先开辟出来一块内存,避免指向内存堆里同一个对象(我是这么理解的,先开辟内存,再把克隆过来的给他)
       try{
           stu = (Student)super.clone();
           //super.clone();调用父类clone方法,父类调用的是本地底层方法了。
       }catch(CloneNotSupportedException e) {
           e.printStackTrace();
       }
       return stu;
   }
}

 public static void main(String[] args) {
        Student stu1 = new Student();
        stu1.setNumber(12345);
        Student stu2 = (Student)stu1.clone();

        System.out.println("学生1:" + stu1.getNumber());
        System.out.println("学生2:" + stu2.getNumber());

        stu2.setNumber(54321);

        System.out.println("学生1:" + stu1.getNumber());
        System.out.println("学生2:" + stu2.getNumber());
    }

深克隆
在深克隆中,无论原型对象的成员变量是值类型还是引用类型,都将复制一份给克隆对象,深克隆将原型对象的所有引用对象也复制一份给克隆对象。
简单来说,在深克隆中,除了对象本身被复制外,对象所包含的所有成员变量(值类型,引用类型)也将复制。
例子说明:

public class Address {
    private String add;
    public void setAdd(String add) {
        this.add = add;
    }
    public String getAdd() {
        return add;
    }
}
public class Student  implements  Cloneable{
    private int number;
    //浅克隆的时候只有值类型,并没有引用类型(Address类型)
    private Address address;
    public int getNumber() {
        return number;
    }

    public void setNumber(int number) {
        this.number = number;
    }

    public void setAddress(Address address) {
        this.address = address;
    }

    public Address getAddress() {
        return address;
    }

    @Override
   public Object clone() {
       Student stu = null;
       try{
           stu = (Student)super.clone();
       }catch(CloneNotSupportedException e) {
           e.printStackTrace();
       }
       return stu;
   }
}

 public static void main(String[] args) {
        Address address=new Address();
        address.setAddress("承德");
        Student stu1 = new Student();
        stu1.setNumber(1);
        stu1.setAddress(address);
        Student stu2 = (Student)stu1.clone();

        System.out.println("学生1:   " + stu1.getNumber()+" 学生1地址:"+stu1.getAddress().getAdd());
        System.out.println("克隆学生: " + stu2.getNumber()+" 学生2地址:"+stu1.getAddress().getAdd());

        stu2.setNumber(2);
        //当我们把地址改了,两个学生的地址都发生了改变。
        //原因是浅复制只是复制了add变量的引用,并没有真正的开辟另一块空间,将值复制后再将引用返回给新对象。
        address.setAdd("三河");

        System.out.println("学生1:   " + stu1.getNumber()+" 学生1地址:"+stu1.getAddress().getAdd());
        System.out.println("克隆学生: " + stu2.getNumber()+" 学生2地址:"+stu1.getAddress().getAdd());
    }

对应的程序输出结果:

在这里插入图片描述
正确做法:

public class Address implements Cloneable {
    private String add;

    public String getAdd() {
        return add;
    }

    public void setAdd(String add) {
        this.add = add;
    }

    @Override
    public Object clone() {
        Address addr = null;
        try {
            addr = (Address) super.clone();
        } catch (CloneNotSupportedException e) {
            e.printStackTrace();
        }
        return addr;
    }
}
public class Student implements Cloneable {
    private int number;
    private Address addr;

    public int getNumber() {
        return number;
    }

    public void setNumber(int number) {
        this.number = number;
    }

    public void setAddress(Address address) {
        this.addr = address;
    }

    public Address getAddress() {
        return addr;
    }

    @Override
    public Object clone() {
        Student stu = null;
        try {
            stu = (Student) super.clone();   //浅复制
        } catch (CloneNotSupportedException e) {
            e.printStackTrace();
        }
        stu.addr = (Address) addr.clone();   //深度复制
        return stu;
    }
}

public class Two {
    public static void main(String[] args) {
        Address address=new Address();
        address.setAdd("承德");
        Student stu1 = new Student();
        stu1.setNumber(1);
        stu1.setAddress(address);
        Student stu2 = (Student)stu1.clone();

        System.out.println("学生1:   " + stu1.getNumber()+" 学生1地址:"+stu1.getAddress().getAdd());
        System.out.println("克隆学生: " + stu2.getNumber()+" 学生2地址:"+stu1.getAddress().getAdd());

        stu2.setNumber(2);
        address.setAdd("三河");

        System.out.println("学生1:   " + stu1.getNumber()+" 学生1地址:"+stu1.getAddress().getAdd());
        System.out.println("克隆学生: " + stu2.getNumber()+" 学生2地址:"+stu2.getAddress().getAdd());
    }
}

自己理解的解释:
错误的浅克隆:
在这里插入图片描述
正确的浅克隆:
在这里插入图片描述
错误的深克隆
在这里插入图片描述
正确的深克隆
在这里插入图片描述

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Javaclone()方法Object定义的方法,它用于创建并返回当前对象的一个副本。这个副本就是一个新的对象,它与原始对象具有相同的属性和方法。通常情况下,我们需要在一个类实现clone()方法来支持对象的克隆克隆对象是在Java处理对象的一种常见方式。通过克隆,我们可以在不修改原始对象的情况下创建该对象的一个副本。这在某些情况下是非常有用的,例如在多线程环境下,我们需要多个线程同时访问同一个对象,但是又不希望它们之间相互干扰。 在Java,要使用clone()方法来实现对象的克隆,我们需要满足两个条件: 1. 实现Cloneable接口:这个接口是一个标记接口,它没有任何方法,只是用来标记一个类可以被克隆。 2. 重写clone()方法:这个方法Object的一个protected方法,需要在我们的类进行重写。在重写这个方法时,我们需要调用super.clone()方法来创建一个新的对象,并将原始对象的属性复制到这个新对象。 下面是一个示例代码,演示了如何在Java实现对象的克隆: ``` public class MyClass implements Cloneable { private int value; public MyClass(int value) { this.value = value; } public int getValue() { return value; } @Override protected Object clone() throws CloneNotSupportedException { return super.clone(); } } ``` 在这个示例代码,我们实现了一个MyClass类,并重写了clone()方法。在这个方法,我们调用了super.clone()方法来创建一个新的对象,并返回这个新对象。由于我们的类实现了Cloneable接口,因此它可以被克隆。 使用这个类进行克隆的代码如下: ``` MyClass obj1 = new MyClass(10); MyClass obj2 = (MyClass) obj1.clone(); System.out.println(obj1.getValue()); // 输出10 System.out.println(obj2.getValue()); // 输出10 ``` 在这个代码,我们创建了一个MyClass对象obj1,并将其克隆为obj2。由于这两个对象具有相同的属性和方法,因此它们的输出结果也是相同的。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值