Java的三个接口Comparable,Comparator,Cloneable(浅拷贝与深拷贝)

2024-06-18 13:44

本文主要是介绍Java的三个接口Comparable,Comparator,Cloneable(浅拷贝与深拷贝),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

Comparable

当我们要进行对象的比较的时候,我们是不能直接用>、< 这些符号直接进行比较的。

由于这是引用类型变量也是自定义类型变量,直接进行比较的时候,我们是通过对象的地址进行比较的我们可以使用==、!= 进行两个对象的地址是否相等,但是不能直接使用 >、< 进行比较,>、< 可以使用在基本的数据类型的比较中,因此 >、< 是不能用于地址的比较的

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述


在Object 类中,我们知道可以使用equals方法来进行对象的比较,返回值是布尔值。如果我们要求返回值是整型的话,我们就要使用到Comparable接口

使用

class Student implements Comparable<Student>{public String name;public int age;public Student(String name, int age) {this.name = name;this.age = age;}@Overridepublic int compareTo(Student o) {return this.age - o.age;}
}

我们需要写上Comparable 接口,后面的<> 里面的内容写你要比较的对象的类型,这里包含泛型的知识,会在数据结构中讲解~~

我们先来看一下Comparable接口:

在这里插入图片描述

Comparable接口中包含 compareTo,因此我们需要重写这个方法,根据不同的比较需求来写不同的比较代码:
在这里插入图片描述
这里是实现age比较,如果是name比较,我们该怎么实现?

由于name是String类,String类有实现Comparable接口的,所以我们直接调用即可~~

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

多个同类型比较

如果我们有很多个学生需要进行比较排序,我们第一时间会想到使用数组来存放,然后通过数组排序(Array.sort())来进行比较排序。

那Array.sort 是怎么进行排序的呢?

事实上Array.sort 是根据待排序的对象中的compareTo方法进行比较的


现在这个类没有Comparable接口:
在这里插入图片描述
我们一运行就会发生异常:

在这里插入图片描述
这是是Student类不能转化为Comparable,说明Array.sort的排序需要该类实现Comparable接口。

Array.sort 会调用 compareTo 方法进行比较。


模拟实现Array.sort(冒泡排序法)

    public static void mysort(Comparable[] comparables) {int flag = 1;for (int i = 0; flag == 1 && i < comparables.length - 1; i++) {flag = 0;for (int j = 0; j < comparables.length - 1 - i; j++) {if(comparables[j].compareTo(comparables[j+1]) > 0) {Comparable tmp = comparables[j];comparables[j] = comparables[j+1];comparables[j+1] = tmp;flag = 1;}}}}

局限

由于compareTo 方法只能重写一次,实现不了重载,因为参数就是所在类的类型(也就意味着这是固定的参数),所以它的局限性就是只能进行一种数值的比较,不能进行多种数值的比较,因此我们一般用在固定的比较,用在默认的比较上,如果要实现不同的数值的比较我们会用到比较器Comparator

Comparator

我们可以使用Comparator实现不同属性比较的类,这里还是以学生类(包括姓名和年龄)作为例子:

public class NameComparator implements Comparator<Student> {@Overridepublic int compare(Student o1, Student o2) {return o1.name.compareTo(o2.name);}
}public class AgeComparator implements Comparator<Student> {@Overridepublic int compare(Student o1, Student o2) {return o1.age - o2.age;}
}

Comparator 后面也需要加上<>,里面填比较的类
还需要重写Comparator里面的compara方法~~

之后我们就可以使用这些比较类的方法了,和类的使用是一样的,先创建对象,再使用里面的方法:

    public static void main(String[] args) {Student stu1 = new Student("zhangsan",10);Student stu2 =new Student("lisi",20);AgeComparator ageComparator = new AgeComparator();int ret = ageComparator.compare(stu1,stu2);System.out.println(ret);NameComparator nameComparator = new NameComparator();ret = nameComparator.compare(stu1,stu2);System.out.println(ret);}

如果你需要使用Array.sort的话,只需要再传比较类就可以了:

        AgeComparator ageComparator = new AgeComparator();Arrays.sort(students,ageComparator);NameComparator nameComparator = new NameComparator();Arrays.sort(students,nameComparator);

Cloneable

当我们需要进行对象的克隆(复制)的时候,我们可以使用clone的接口,这是Object类的,在上一篇文章我们就提到其中的三个方法,现在我们就来将克隆方法(clone)

在这里插入图片描述

使用

在这里插入图片描述

我们需要重写clone方法
在这里插入图片描述
因为clone方法这是protected修饰的,只能在同一个包下访问或者子类自己能访问,在Test类下是无法访问的,所以我们只能通过重写clone方法。

快捷键如下:
在这里插入图片描述
在这里插入图片描述

编译器会帮我们生成如下的代码:

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

在这里插入图片描述

强制类型转换,因为clone方法的放回值是Object类型的,我们需要强制类型转换为Student

Student stu2 = (Student)stu1.clone();

在这里插入图片描述

处理异常,异常我会在后续的文章中讲解,我们需要在调用clone方法的方法旁边写上throws CloneNotSupportedException

public static void main(String[] args) throws CloneNotSupportedException{Student stu1 = new Student("zhangsan",14);Student stu2 = (Student)stu1.clone();}

但是当我们运行的时候会发现下面的异常:
在这里插入图片描述

这里写的是不支持clone,这时候我们就需要写上Cloneable接口,这是一个空的接口,目的就是来标记这个类是支持克隆的~~

在这里插入图片描述

public class Student implements Cloneable

完成上述步骤我们就可以实现克隆了:

在这里插入图片描述

在这里插入图片描述


当我们需要克隆的对象里面还包含一个对象的时候,如果我们不拷贝这个被包含的对象,那这就是浅拷贝,如果需要拷贝多一份新的被包含的对象时,那就是深拷贝。

以下面的代码为例:
在这里插入图片描述
在这里插入图片描述
我们来克隆一个per1:

Person per1 = new Person("zhagnsan",10);

浅拷贝示意图:
在这里插入图片描述

深拷贝示意图:
在这里插入图片描述

浅拷贝

浅拷贝的实现和上面普通拷贝的实现是一样的,这里不赘述了,只有深拷贝有一些不一样

通过上面的示意图,我们来做一下题目,说明下面的运行结果:

public class Money {public double m = 9.99;
}public class Person implements Cloneable{public String name;public int age;public Money money = new Money();public Person(String name, int age) {this.name = name;this.age = age;}@Overridepublic String toString() {return "Person{" +"name='" + name + '\'' +", age=" + age +", money=" + money +'}';}@Overrideprotected Object clone() throws CloneNotSupportedException {return super.clone();}
}class Test{public static void main(String[] args) throws CloneNotSupportedException{Person per1 = new Person("zhagnsan",10);Person per2 = (Person) per1.clone();per2.money.m = 6.6;System.out.println("per1:"+per1.money.m);System.out.println("per2:"+per2.money.m);}
}

在这里插入图片描述

在这里插入图片描述

这里显而易见,浅拷贝后per1和per2是共享money的,所以有一个人的money发生改变,另一个人的money也会发生改变。

深拷贝

我们知道深拷贝需要再拷贝多一份全新的被包含的对象,所以我们需要实现被包含的对象的拷贝:

public class Money implements Cloneable{public double m = 9.99;@Overrideprotected Object clone() throws CloneNotSupportedException {return super.clone();}
}

这是第一步,下一步我们需要在包含该类的类中的clone方法调用Money中的clone方法,才能实现完整的深拷贝工作:

所以我们要修改Person中的clone方法:

    @Overrideprotected Object clone() throws CloneNotSupportedException {Person tmp = (Person) super.clone();tmp.money= (Money) this.money.clone();return tmp;}

现在再思考一下,下面的代码运行结果是什么?

public class Money implements Cloneable{public double m = 9.99;@Overrideprotected Object clone() throws CloneNotSupportedException {return super.clone();}
}public class Person implements Cloneable{public String name;public int age;public Money money = new Money();public Person(String name, int age) {this.name = name;this.age = age;}@Overridepublic String toString() {return "Person{" +"name='" + name + '\'' +", age=" + age +", money=" + money +'}';}@Overrideprotected Object clone() throws CloneNotSupportedException {Person tmp = (Person) super.clone();tmp.money= (Money) this.money.clone();return tmp;}
}class Test{public static void main(String[] args) throws CloneNotSupportedException{Person per1 = new Person("zhagnsan",10);Person per2 = (Person) per1.clone();per2.money.m = 6.6;System.out.println("per1:"+per1.money.m);System.out.println("per2:"+per2.money.m);}
}

在这里插入图片描述

在这里插入图片描述

深拷贝已经重新将Money拷贝多一份了,所以per2的money改变了并不会影响到per1的money.

这篇关于Java的三个接口Comparable,Comparator,Cloneable(浅拷贝与深拷贝)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1072162

相关文章

Spring Boot @RestControllerAdvice全局异常处理最佳实践

《SpringBoot@RestControllerAdvice全局异常处理最佳实践》本文详解SpringBoot中通过@RestControllerAdvice实现全局异常处理,强调代码复用、统... 目录前言一、为什么要使用全局异常处理?二、核心注解解析1. @RestControllerAdvice2

Spring IoC 容器的使用详解(最新整理)

《SpringIoC容器的使用详解(最新整理)》文章介绍了Spring框架中的应用分层思想与IoC容器原理,通过分层解耦业务逻辑、数据访问等模块,IoC容器利用@Component注解管理Bean... 目录1. 应用分层2. IoC 的介绍3. IoC 容器的使用3.1. bean 的存储3.2. 方法注

Spring事务传播机制最佳实践

《Spring事务传播机制最佳实践》Spring的事务传播机制为我们提供了优雅的解决方案,本文将带您深入理解这一机制,掌握不同场景下的最佳实践,感兴趣的朋友一起看看吧... 目录1. 什么是事务传播行为2. Spring支持的七种事务传播行为2.1 REQUIRED(默认)2.2 SUPPORTS2

怎样通过分析GC日志来定位Java进程的内存问题

《怎样通过分析GC日志来定位Java进程的内存问题》:本文主要介绍怎样通过分析GC日志来定位Java进程的内存问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、GC 日志基础配置1. 启用详细 GC 日志2. 不同收集器的日志格式二、关键指标与分析维度1.

Java进程异常故障定位及排查过程

《Java进程异常故障定位及排查过程》:本文主要介绍Java进程异常故障定位及排查过程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、故障发现与初步判断1. 监控系统告警2. 日志初步分析二、核心排查工具与步骤1. 进程状态检查2. CPU 飙升问题3. 内存

java中新生代和老生代的关系说明

《java中新生代和老生代的关系说明》:本文主要介绍java中新生代和老生代的关系说明,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、内存区域划分新生代老年代二、对象生命周期与晋升流程三、新生代与老年代的协作机制1. 跨代引用处理2. 动态年龄判定3. 空间分

Java设计模式---迭代器模式(Iterator)解读

《Java设计模式---迭代器模式(Iterator)解读》:本文主要介绍Java设计模式---迭代器模式(Iterator),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,... 目录1、迭代器(Iterator)1.1、结构1.2、常用方法1.3、本质1、解耦集合与遍历逻辑2、统一

Java内存分配与JVM参数详解(推荐)

《Java内存分配与JVM参数详解(推荐)》本文详解JVM内存结构与参数调整,涵盖堆分代、元空间、GC选择及优化策略,帮助开发者提升性能、避免内存泄漏,本文给大家介绍Java内存分配与JVM参数详解,... 目录引言JVM内存结构JVM参数概述堆内存分配年轻代与老年代调整堆内存大小调整年轻代与老年代比例元空

深度解析Java DTO(最新推荐)

《深度解析JavaDTO(最新推荐)》DTO(DataTransferObject)是一种用于在不同层(如Controller层、Service层)之间传输数据的对象设计模式,其核心目的是封装数据,... 目录一、什么是DTO?DTO的核心特点:二、为什么需要DTO?(对比Entity)三、实际应用场景解析

Java 线程安全与 volatile与单例模式问题及解决方案

《Java线程安全与volatile与单例模式问题及解决方案》文章主要讲解线程安全问题的五个成因(调度随机、变量修改、非原子操作、内存可见性、指令重排序)及解决方案,强调使用volatile关键字... 目录什么是线程安全线程安全问题的产生与解决方案线程的调度是随机的多个线程对同一个变量进行修改线程的修改操