【Java】Clonable 接口

简介: 【Java】Clonable 接口

如何克隆一个引用所指的对象呢?首先一个前提,他是可克隆的,我们要实现一个Clonable 接口。我们来看一个这个接口:



可以发现里面是空的,我们把这种空接口叫做标记接口,作用就是表示当前对象是可以被克隆的。

我们要实现克隆,首先要在类中重写Object的clone方法:


package csdn;
 
import java.lang.reflect.Array;
import java.util.Arrays;
 
class  student implements  Cloneable{
    public  String name;
    public student(String name) {
        this.name = name;
    }
 
    @Override
    public String toString() {
        return "student{" +
                "name='" + name + '\'' +
                '}';
    }
 
    @Override
    protected Object clone() throws CloneNotSupportedException {  //重写父类Object的clone方法
        return super.clone();
    }
}
public class test2 {
    public static void main(String[] args) throws CloneNotSupportedException{
        student student1=new student("张三");
        student student2= (student)student1.clone();  //实现克隆,然后向下转型
        System.out.println(student1);
        System.out.println(student2);
 
 
    }
}


这样就实现了简单的克隆。

那我们再考虑另一种情况,我们来看这一段代码:


package csdn;
 
import com.sun.scenario.effect.impl.sw.sse.SSEBlend_SRC_OUTPeer;
 
import java.lang.reflect.Array;
import java.util.Arrays;
 class Money{
     double money=12.25;
 
 }
class  student implements  Cloneable{
    public  String name;
    public  Money  m=new Money();
 
    public student(String name) {
        this.name = name;
    }
 
 
    @Override
    protected Object clone() throws CloneNotSupportedException {  //重写父类Object的clone方法
        return super.clone();
    }
}
public class test2 {
    public static void main(String[] args) throws CloneNotSupportedException{
        student student1=new student("张三");
        student student2= (student)student1.clone();  //实现克隆,然后向下转型
        System.out.println(student1.m.money);
        System.out.println(student2.m.money);
        student2.m.money=99;
        System.out.println("==============");
        System.out.println(student1.m.money);
        System.out.println(student2.m.money);
 
    }
}


理论上来说,给克隆的对象改了值应该不会影响原对象的money值,我们这都是克隆出去的,但是当我们运行的时候却发现原来对象的money值也被修改了(这种拷贝方式叫做浅拷贝):


这跟我们的需求就不一样,这是什么原因呢?这是因为我们在克隆的时候没有克隆Money对象。



(糟糕的画图技术,希望大家能看明白)

我们解决方法是把Money也克隆出来就可以了,Money也必须支持克隆,也就是实现Clonable 接口。



然后我们再来修改一下student类里面的克隆方法:



此时我们再来看一下运行结果,就跟之前不一样了:


此时就达到了我们的深拷贝。所以如果要实现深拷贝,你当前对象里面的每一个对象都得克隆。

代码:

package csdn;
 
import com.sun.scenario.effect.impl.sw.sse.SSEBlend_SRC_OUTPeer;
 
import java.lang.reflect.Array;
import java.util.Arrays;
 class Money implements Cloneable{
     double money=12.25;
 
     @Override
     protected Object clone() throws CloneNotSupportedException {  //重写父类Object的clone方法
         return super.clone();
     }
 }
class  student implements  Cloneable{
    public  String name;
    public  Money  m=new Money();
 
    public student(String name) {
 
        this.name = name;
    }
 
    @Override
    protected Object clone() throws CloneNotSupportedException {  //重写父类Object的clone方法
        student student=(student)super.clone(); //只是克隆了student对象
        student.m=(Money)this.m.clone();  //克隆了student对象里面的Money对象
        return student;
    }
}
public class test2 {
    public static void main(String[] args) throws CloneNotSupportedException{
        student student1=new student("张三");
        student student2= (student)student1.clone();  //实现克隆,然后向下转型
        System.out.println(student1.m.money);
        System.out.println(student2.m.money);
        student2.m.money=99;
        System.out.println("==============");
        System.out.println(student1.m.money);
        System.out.println(student2.m.money);
 
    }
}

 


目录
相关文章
|
20天前
|
算法 Java 数据处理
从HashSet到TreeSet,Java集合框架中的Set接口及其实现类以其“不重复性”要求,彻底改变了处理唯一性数据的方式。
从HashSet到TreeSet,Java集合框架中的Set接口及其实现类以其“不重复性”要求,彻底改变了处理唯一性数据的方式。HashSet基于哈希表实现,提供高效的元素操作;TreeSet则通过红黑树实现元素的自然排序,适合需要有序访问的场景。本文通过示例代码详细介绍了两者的特性和应用场景。
34 6
|
20天前
|
存储 Java 数据处理
Java Set接口凭借其独特的“不重复”特性,在集合框架中占据重要地位
【10月更文挑战第16天】Java Set接口凭借其独特的“不重复”特性,在集合框架中占据重要地位。本文通过快速去重和高效查找两个案例,展示了Set如何简化数据处理流程,提升代码效率。使用HashSet可轻松实现数据去重,而contains方法则提供了快速查找的功能,彰显了Set在处理大量数据时的优势。
31 2
|
1天前
|
Java
java线程接口
Thread的构造方法创建对象的时候传入了Runnable接口的对象 ,Runnable接口对象重写run方法相当于指定线程任务,创建线程的时候绑定了该线程对象要干的任务。 Runnable的对象称之为:线程任务对象 不是线程对象 必须要交给Thread线程对象。 通过Thread的构造方法, 就可以把任务对象Runnable,绑定到Thread对象中, 将来执行start方法,就会自动执行Runable实现类对象中的run里面的内容。
9 1
|
6天前
|
Java 开发者
在Java多线程编程的世界里,Lock接口正逐渐成为高手们的首选,取代了传统的synchronized关键字
在Java多线程编程的世界里,Lock接口正逐渐成为高手们的首选,取代了传统的synchronized关键字
32 4
|
13天前
|
安全 Java
在 Java 中使用实现 Runnable 接口的方式创建线程
【10月更文挑战第22天】通过以上内容的介绍,相信你已经对在 Java 中如何使用实现 Runnable 接口的方式创建线程有了更深入的了解。在实际应用中,需要根据具体的需求和场景,合理选择线程创建方式,并注意线程安全、同步、通信等相关问题,以确保程序的正确性和稳定性。
|
11天前
|
Java
Java基础(13)抽象类、接口
本文介绍了Java面向对象编程中的抽象类和接口两个核心概念。抽象类不能被实例化,通常用于定义子类的通用方法和属性;接口则是完全抽象的类,允许声明一组方法但不实现它们。文章通过代码示例详细解析了抽象类和接口的定义及实现,并讨论了它们的区别和使用场景。
|
11天前
|
Java 测试技术 API
Java零基础-接口详解
【10月更文挑战第19天】Java零基础教学篇,手把手实践教学!
16 1
|
16天前
|
Java 开发者
在Java多线程编程中,创建线程的方法有两种:继承Thread类和实现Runnable接口
【10月更文挑战第20天】在Java多线程编程中,创建线程的方法有两种:继承Thread类和实现Runnable接口。本文揭示了这两种方式的微妙差异和潜在陷阱,帮助你更好地理解和选择适合项目需求的线程创建方式。
13 3
|
16天前
|
Java
在Java多线程编程中,实现Runnable接口通常优于继承Thread类
【10月更文挑战第20天】在Java多线程编程中,实现Runnable接口通常优于继承Thread类。原因包括:1) Java只支持单继承,实现接口不受此限制;2) Runnable接口便于代码复用和线程池管理;3) 分离任务与线程,提高灵活性。因此,实现Runnable接口是更佳选择。
28 2
|
16天前
|
Java
Java中多线程编程的基本概念和创建线程的两种主要方式:继承Thread类和实现Runnable接口
【10月更文挑战第20天】《JAVA多线程深度解析:线程的创建之路》介绍了Java中多线程编程的基本概念和创建线程的两种主要方式:继承Thread类和实现Runnable接口。文章详细讲解了每种方式的实现方法、优缺点及适用场景,帮助读者更好地理解和掌握多线程编程技术,为复杂任务的高效处理奠定基础。
27 2
下一篇
无影云桌面