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.

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/diannao/29447.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

解决 pdf.js 出现 TypeError: key.split(...).at is not a function 报错问题

问题 使用pdf.js v2版本部分机型出现 TypeError: key.split(…).at is not a function 报错 解决 参考 https://github.com/wojtekmaj/react-pdf/issues/1465 方式一&#xff08;推荐&#xff09; import core-js/features/array/at;方式二 (解决了部分机型浏览器问题) if …

Spring Cloud 专题-前言篇(1)

引言 随着微服务架构的兴起&#xff0c;Spring Cloud 作为一套基于 Spring Boot 实现的云应用开发工具集&#xff0c;为开发者提供了在分布式系统&#xff08;如配置管理、服务发现、断路器、智能路由、微代理、控制总线等&#xff09;中快速构建一些常见模式的能力。本篇文档…

2024年大韩民国最佳品牌大赏 彭雨凡荣获“海外邀请特别奖”

14日&#xff0c;“2024年大韩民国最佳品牌大赏-韩流演艺大赏”颁奖典礼在韩国首尔永登浦区汝矣岛洞国会议员会馆第2会议室举办。 演员彭雨凡荣获“海外邀请特别奖”。 据悉&#xff0c;由大韩民国最佳品牌协会和世宗大王国民委员会&#xff08;理事长 LEE YUNTAE&#xff09…

关于IOMMU问题的扩展

关联CSDN&#xff1a; Steam Deck OLED WLAN下载速率过低问题的排查和解决-CSDN博客 前言 如前所述&#xff0c;Steam Deck OLED WLAN速率低问题和IOMMU有一定的关系&#xff0c;这里我们对IOMMU为什么会对速率有影响进行一个较深入的理解。 对于IOMMU我相信大家通过网上的…

模板引擎与 XSS 防御

在 View 层&#xff0c;可以解决 XSS 问题。在本书的“跨站脚本攻击”一章中&#xff0c;阐述了“输入检查” 与“输出编码”这两种方法在 XSS 防御效果上的差异。XSS 攻击是在用户的浏览器上执行的&#xff0c; 其形成过程则是在服务器端页面渲染时&#xff0c;注入了恶意的 H…

Android中的Audio系统框架分析(一)

概述 Audio系统是Android 平台重要的组成部分&#xff0c;我们将从以下几个方面来讲解&#xff1a; 一Audio基础知识讲解 二、Android系统中Audio框架 Audio基础知识讲解 我们大家知道声音是由物体振动产生的声波。是通过介质&#xff08;空气或固体、液体&#xff09;传播并…

CrossOver Games For Mac官方下载_2024电脑最新版软件安装包下载

CrossOver Pro For Mac是由codewaver公司开发的类虚拟机软件&#xff0c;目的是使linux和Mac OS X操作系统和window系统兼容。CrossOver Pro For Mac能够直接在Mac上运行Windows软件与游戏&#xff0c;而不需虚拟机&#xff0c;功能是非常强大的&#xff0c;值得大家下载使用。…

Linux系统基本知识----1

1.什么是Linux中的权限&#xff0c;并举例说明。 ****Linux权限是操作系统用来控制特定用户或用户组可以对文件或目录执行的操作的一种机制。例如&#xff0c;chmod 755 filename 命令会设置filename文件的权限&#xff0c;使得拥有者可以读/写/执行&#xff0c;而组用户和其他…

Android Audio实战——声道信息回调(五)

在前面的 AudioTrack 构造中,我们传入了音频的声道信息,这一节我们就来详细介绍一下声道的配置信息。 一、声道介绍 音频中的声道配置从单声道到双声道(立体声)、再到多声道系统(如5.1和7.1),代表了声音录制和回放技术的发展,旨在提供越来越丰富和沉浸式的听觉体验。 …

在Spring Boot中使用Sa-Token实现路径拦截和特定接口放行

在Spring Boot中使用Sa-Token实现路径拦截和特定接口放行 很喜欢的一段话&#xff1a;别想太多&#xff0c;好好生活&#xff0c;也许日子过着过着就会有答案&#xff0c;努力走着走着就会有温柔的着落。 春在路上&#xff0c;花在枝上&#xff0c;所有的美好都在路上&#xff…

【测试专题】系统测试报告(原件Word)

软件测试报告在软件开发过程中起着至关重要的作用&#xff0c;主要有以下几个主要原因&#xff1a; 1、确保软件质量 2、提供决策支持 3、记录测试过程和结果 4、促进沟通和协作 5、符合标准和法规要求 6、改进测试流程和策略 7、降低风险 软件开发全套资料获取进主页或者本文末…

程序猿大战Python——文件操作、异常、模块——os模块

查看目录 目标&#xff1a;了解查看目录的使用。 Python中的os模块包含有操作系统所具备的功能&#xff0c;如查看路径、创建目录、显示文件列表等。 os模块是Python标准库&#xff0c;可直接导入使用&#xff1a; # 导入os模块 import os 在Python中&#xff0c;os模块的常…

全面介绍Linux中的Vim编辑器

一、Vim简介 Vim&#xff08;Vi IMproved&#xff09;是从经典的Unix文本编辑器Vi发展而来的一个强大、可扩展的文本编辑器。Vim被设计为高度可配置的&#xff0c;并且具备强大的插件系统&#xff0c;使得它不仅适用于程序员&#xff0c;还适合日常文本编辑。与普通的文本编辑…

H5应用调用企业微信扫一扫API扫码方法

场景&#xff1a;自行开发的企微应用例如扫码入库、二维码资产盘点等等 官网地址:使用说明 - 接口文档 - 企业微信开发者中心 扫一扫接口说明:企业微信扫一扫 - 接口文档 - 企业微信开发者中心 使用Js版本: http://res.wx.qq.com/open/js/jweixin-1.2.0.js 前端JS代码&…

IO流(二)

IO流&#xff08;二&#xff09; 目录 IO流 —— 字符流IO流 —— 缓冲流IO流 —— 转换流IO流 —— 打印流IO流 —— 数据流IO流 —— 序列化流 1.IO流 —— 字符流 文件字符输入流 —— 读字符数据进来 字节流&#xff1a;适合复制文件等&#xff0c;不适合读写文本文件字…

nginx rewrite地址重写

目录 常用的nginx正则表达式 location和rewrite的区别 一、location 1.location常用匹配类型 2.location匹配机制 3.实际工作中三大匹配规则 1.网站首页匹配 2.网站静态页面&#xff0c;通过前缀匹配或通用匹配在nginx服务器本地处理 3.网站动态页面&#xff0c;通过匹…

TypeScript语法解析与进阶扩展

TypeScript 1、类型别名2、字符串字面量类型3、元组4、枚举5、类5.1、public private protected5.2、readonly5.3、抽象类5.4、静态成员5.5、类实现接口5.6、接口继承接口5.7、接口继承类 6、泛型6.1、多个类型参数6.2、泛型约束6.3、泛型接口6.4、泛型类6.5、泛型参数的默认类…

Veeam Backup Enterprise Manager身份验证绕过漏洞(CVE-2024-29849)

一、漏洞概述【漏洞通告】 漏洞名称 Veeam Backup Enterprise Manager身份验证绕过漏洞 CVE ID CVE-2024-29849 漏洞类型 身份验证绕过 发现时间 2024-05-22 漏洞评分 9.8 漏洞等级 严重 攻击向量 网络 所需权限 无 利用难度 低 用户交互 无 PoC/EXP 已…

PostgreSQL源码分析——initdb

数据库初始化 在安装完数据库后&#xff0c;需要进行初始化数据库操作&#xff0c;对应PostgreSQL数据库中就是需要进行initdb后&#xff0c;才能对数据库进行启动。initdb的过程&#xff0c;其实就是创建数据库实例的过程&#xff0c;生成模板数据库和相应的目录、文件信息&a…

uniapp小程序限制微信群访问(图文教程)

我有一个微信小程序 “程序员实用资源” 我现在只想让我的微信群可以访问这个小程序的所有功能 所以我必须对我小程序的来源进行限制&#xff0c;让部分功能在正常访问的时候提示没有加群&#xff0c;不可访问&#xff0c;只有从群内点击进入小程序的时候才可以访问这部分功能…