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(浅拷贝与深拷贝)
Comparable 当我们要进行对象的比较的时候,我们是不能直接用>、< 这些符号直接进行比较的。 由于这是引用类型变量也是自定义类型变量,直接进行比较的时候,我们是通过对象的地址进行比较的,我们可以使用、! 进行两个对象的…...
pytorch学习笔记7
getitem在进行索引取值的时候自动调用,也是一个魔法方法,就像列表索引取值那样,一个意思 import torchvision from torch.utils.data import DataLoaderdata_transformtorchvision.transforms.Compose([torchvision.transforms.ToTensor()] ) test_datatorchvision.datasets.C…...
LeetCode热题3.无重复的最长字串
前言: 经过前序的一系列数据结构和算法学习后,开始用leetCode热题练练手。 . - 力扣(LeetCode) 给定一个字符串 s ,请你找出其中不含有重复字符的最长子串的长度。 示例 1: 输入: s "abcabcbb" 输出: 3 解释: 因为…...
Python武器库开发-武器库篇之SQL注入扫描器(五十九)
Python武器库开发-武器库篇之SQL注入扫描器(五十九) SQL注入漏洞简介以及危害 SQL注入漏洞是一种常见的Web应用程序漏洞,攻击者可以利用该漏洞在应用程序的数据库中执行恶意的SQL查询或指令。这可能导致数据泄露、数据损坏、应用程序崩溃或未经授权的访问。 SQL注…...
图说设计模式:单例模式
更多C学习笔记,关注 wx公众号:cpp读书笔记 5. 单例模式 单例模式 模式动机模式定义模式结构时序图代码分析模式分析实例优点缺点适用环境模式应用模式扩展总结 5.1. 模式动机 对于系统中的某些类来说,只有一个实例很重要,例如…...
探索设计模式——单例模式详解
前言:设计模式的作用主要是为了——利用设计方式的重用来自动地提高代码的重新利用、提高代码的灵活性、节省时间, 提高开发效率、低耦合,封装特性显著, 接口预留有利于扩展。 设计模式的种类有很多种,本篇内容主要讲解…...
建筑垃圾/城市固废倾倒转移乱象:EasyCVR+AI智能视频监控方案助力城市环保监管
近日有新闻记者报道,中央生态环境保护督察组在上海、浙江、江西、湖北、湖南、重庆、云南7省市督察发现,一些地方建筑垃圾处置工作存在明显短板,乱堆乱倒问题时有发生,比如,江西湘东区在杨家田地块违规设置弃土场&…...
C的I/O操作
目录 引言 一、文件与目录操作 1. 打开与关闭文件 2. 文件读写操作 3. 文件定位与错误处理 二、字符流与字节流 1. 字符流处理 2. 字节流处理 三、序列化与反序列化 1. 序列化 2. 反序列化 四、新的I/O(NIO) 表格总结 文件与目录操作 字符…...
Android Audio实战——声道信息回调(五)
在前面的 AudioTrack 构造中,我们传入了音频的声道信息,这一节我们就来详细介绍一下声道的配置信息。 一、声道介绍 音频中的声道配置从单声道到双声道(立体声)、再到多声道系统(如5.1和7.1),代表了声音录制和回放技术的发展,旨在提供越来越丰富和沉浸式的听觉体验。 …...
ThreeJS给模型添加介绍文字(贴在模型上 不会一直面向我们)
使用到 FontLoader跟 TextGeometry 引包 import {TextGeometry} from "three/examples/jsm/geometries/TextGeometry"; import {FontLoader} from "three/examples/jsm/loaders/FontLoader";使用 // 创建字体加载器并加载字体 const fontLoader new Fo…...
[Qt] Qt Creator 以及 Qt 在线安装教程
一、Qt Creator 下载及安装 1、从以下镜像源下载安装包常规安装即可 Qt Creator 也可以在第二步Qt 在线安装时一次性勾选安装,见后文 Qt Creator 中科大源下载地址 二、Qt 在线安装 1、根据所在平台选择对应的安装器下载 Qt 在线安装器下载 2、可能的安装报错…...
【大分享05】动态容差归档,打通不动产登记管理“最后一公里”
关注我们 - 数字罗塞塔计划 - 本篇是参加由电子文件管理推进联盟联合数字罗塞塔计划发起的“大分享”活动投稿文章,来自上海涵妍档案信息技术有限责任公司,作者:陈雪。 一、政策背景 在“互联网政务服务”的浪潮下,各级政府机构…...
嵌入式模拟电路面试题大全及参考答案(持续更新)
目录 理想运算放大器的两个基本特性 共模抑制比(CMRR)及其重要性 负反馈在放大器中的作用 差分放大电路的工作原理 使用运算放大器构建非反相放大器 电源抑制比(PSRR) 带宽(BW)在放大器中的含义 计算RC低通滤波器的截止频率 基本的积分电路及其时间常数 增益-带…...
【C语言】解决C语言报错:Uninitialized Variable
文章目录 简介什么是Uninitialized VariableUninitialized Variable的常见原因如何检测和调试Uninitialized Variable解决Uninitialized Variable的最佳实践详细实例解析示例1:局部变量未初始化示例2:数组未初始化示例3:指针未初始化示例4&am…...
RabbitMQ实践——交换器(Exchange)绑定交换器
在《RabbitMQ实践——交换器(Exchange)和绑定(Banding)》一文中,我们实验了各种交换器。我们可以把交换器看成消息发布的入口,而消息路由规则则是由“绑定关系”(Banding)来定义&…...
使用 Vue 官方脚手架初始化 Vue3 项目
Vite 官网:https://cn.vitejs.dev/ Vue 官网:https://vuejs.org/ Vue 官方文档:https://cn.vuejs.org/guide/introduction.html Element Plus 官网:https://element-plus.org/ Tailwind CSS 官网:https://tailwindcss.…...
C语言中的宏定义(#define)和函数调用的区别
C语言中的宏定义(#define)和函数调用在概念、工作方式以及它们对代码的影响上有显著的区别。以下是它们之间的主要差异: 宏定义(#define) 工作方式:宏定义是在预处理阶段进行的文本替换。预处理器会在编译…...
196. 删除重复的电子邮箱
196. 删除重复的电子邮箱 题目链接:196. 删除重复的电子邮箱 代码如下: # Write your MySQL query statement below delete from Person as p where p.id not in(select e.id from (select min(id) as idfrom Person group by email ) as e )...
Android 大话binder通信 (上)
戳蓝字“牛晓伟”关注我哦! 用心坚持输出易读、有趣、有深度、高质量、体系化的技术文章 本文摘要 用故事的方式把binder通信的整个过程都描述出来,binder通信都经历了哪些节点,在这些节点上的数据有哪些变化,同时还对binder通…...
DevOps学习回顾01-技能发展路线-岗位能力-体系认知
事为先,人为重–事在人为 参考来源: 极客时间专栏:DevOps实战笔记,作者:石雪峰 课程链接:https://time.geekbang.org/column/intro/235 时代的典型特征 VUCA VUCA 是指易变性(Volatility&…...
日语学习-日语知识点小记-构建基础-JLPT-N4阶段(33):にする
日语学习-日语知识点小记-构建基础-JLPT-N4阶段(33):にする 1、前言(1)情况说明(2)工程师的信仰2、知识点(1) にする1,接续:名词+にする2,接续:疑问词+にする3,(A)は(B)にする。(2)復習:(1)复习句子(2)ために & ように(3)そう(4)にする3、…...
YSYX学习记录(八)
C语言,练习0: 先创建一个文件夹,我用的是物理机: 安装build-essential 练习1: 我注释掉了 #include <stdio.h> 出现下面错误 在你的文本编辑器中打开ex1文件,随机修改或删除一部分,之后…...
令牌桶 滑动窗口->限流 分布式信号量->限并发的原理 lua脚本分析介绍
文章目录 前言限流限制并发的实际理解限流令牌桶代码实现结果分析令牌桶lua的模拟实现原理总结: 滑动窗口代码实现结果分析lua脚本原理解析 限并发分布式信号量代码实现结果分析lua脚本实现原理 双注解去实现限流 并发结果分析: 实际业务去理解体会统一注…...
Rust 异步编程
Rust 异步编程 引言 Rust 是一种系统编程语言,以其高性能、安全性以及零成本抽象而著称。在多核处理器成为主流的今天,异步编程成为了一种提高应用性能、优化资源利用的有效手段。本文将深入探讨 Rust 异步编程的核心概念、常用库以及最佳实践。 异步编程基础 什么是异步…...
大数据学习(132)-HIve数据分析
🍋🍋大数据学习🍋🍋 🔥系列专栏: 👑哲学语录: 用力所能及,改变世界。 💖如果觉得博主的文章还不错的话,请点赞👍收藏⭐️留言Ǵ…...
springboot整合VUE之在线教育管理系统简介
可以学习到的技能 学会常用技术栈的使用 独立开发项目 学会前端的开发流程 学会后端的开发流程 学会数据库的设计 学会前后端接口调用方式 学会多模块之间的关联 学会数据的处理 适用人群 在校学生,小白用户,想学习知识的 有点基础,想要通过项…...
腾讯云V3签名
想要接入腾讯云的Api,必然先按其文档计算出所要求的签名。 之前也调用过腾讯云的接口,但总是卡在签名这一步,最后放弃选择SDK,这次终于自己代码实现。 可能腾讯云翻新了接口文档,现在阅读起来,清晰了很多&…...
【JVM】Java虚拟机(二)——垃圾回收
目录 一、如何判断对象可以回收 (一)引用计数法 (二)可达性分析算法 二、垃圾回收算法 (一)标记清除 (二)标记整理 (三)复制 (四ÿ…...
华为OD机试-最短木板长度-二分法(A卷,100分)
此题是一个最大化最小值的典型例题, 因为搜索范围是有界的,上界最大木板长度补充的全部木料长度,下界最小木板长度; 即left0,right10^6; 我们可以设置一个候选值x(mid),将木板的长度全部都补充到x,如果成功…...
【前端异常】JavaScript错误处理:分析 Uncaught (in promise) error
在前端开发中,JavaScript 异常是不可避免的。随着现代前端应用越来越多地使用异步操作(如 Promise、async/await 等),开发者常常会遇到 Uncaught (in promise) error 错误。这个错误是由于未正确处理 Promise 的拒绝(r…...

