javaSE - 三个常用的接口(Comparable,Comparator,Cloneable)

news2025/1/17 3:12:17

1、Comparable

英 [ˈkɒmpərəbl] 美 [ˈkɑːmpərəbl]
可比较的;可比的;可比性;可比;可比较

2、Comparator
美 [kəmˈpɜrətər]
n.
比较器,比色器,比较电路,比长仪,场强计

3、 Cloneable

可复制的

一、Comparable 接口

比如说:对一个整型数组排序

public static void main(String[] args) {
        int[] array = {15,20,89,4,10,47,36};
        //排序前
        System.out.println(Arrays.toString(array));
        Arrays.sort(array);
        //排序后
        System.out.println(Arrays.toString(array));
    }

在这里插入图片描述
但是这种排序都是比较简单的排序,是针对整型,浮点型…

如果是对一些复杂类型进行排序
比如说:对学生进行排序
学生有很多属性,名字,年龄,分数等等

class Student{
    public String name;
    public int age;
    public double score;

    public Student(String name, int age, double score) {
        this.name = name;
        this.age = age;
        this.score = score;
    }

    @Override
    public String toString() {
        return "Student{" +
                "name='" + name + '\'' +
                ", age=" + age +
                ", score=" + score +
                '}';
    }
}

public class TestDome {

    public static void main(String[] args) {
        Student[] students = new Student[3];
        students[0] = new Student("张三", 18, 58.5);
        students[1] = new Student("李四", 10, 90);
        students[2] = new Student("王二", 50, 70.7);
        //排序前
        System.out.println(Arrays.toString(students));
        Arrays.sort(students);
        //排序后
        System.out.println(Arrays.toString(students));
    }
}

在这里插入图片描述
上面并没有告诉我们比较的规则,是指定根据什么来排序,是根据姓名,还是年龄,还是分数来进行排序,是升序还是降序???
这样应该怎样进行排序呢???

下面我们看一下sort的实现,是怎样实现的

在这里插入图片描述
就是让每个元素都去调用 Comparable。Comparable就是 数组按照某个规律去排序。

comparable 是一个接口,这个接口里面只有一个compareTo方法
在这里插入图片描述

所以我们在实现学生这个类的时候就要实现 comparable 接口,重写compareTo方法
compareTo 方法是用来比较数组,那就需要看比较 结果是大于0,还是等于0,再者小于0.
在这里插入图片描述

class Student implements Comparable<Student>{
    public String name;
    public int age;
    public double score;

    public Student(String name, int age, double score) {
        this.name = name;
        this.age = age;
        this.score = score;
    }

    @Override
    public String toString() {
        return "Student{" +
                "name='" + name + '\'' +
                ", age=" + age +
                ", score=" + score +
                '}';
    }

    //this,谁调用compareTo 谁就是这个this
    //这个compareTo 方法就是根据学生的年龄进行比较
    @Override
    public int compareTo(Student o) {
//        if(this.age > o.age){
//            return 1;
//        }else if(this.age == o.age){
//            return 0;
//        }else {
//            return -1;
//        }
        return this.age - o.age;
    }
}

在这里插入图片描述
那此时就可以将学生进行排序了,根据学生的年龄来排序
在这里插入图片描述
由上得知 Arrays.sort();默认排序规律是升序(从小到大),那么逆序怎么实现呢?(交换 this.age 与 o.age的位置就可以了)
在这里插入图片描述

自定义的数据类型,要进行大小的比较,一定要实现可以比较的接口Comparable

但是 Comparable 接口存在着缺陷: 如果我们要以学生的分数或者名字来排序,就需要重新 把 compareTo 方法的实现重写一遍,

   public int compareTo(Student o) {
        return (int)(this.score - o.score);
    }

那么通过名字去排序,要怎么写。
在这里插入图片描述
调用String的compareTo方法来进行比较

   public int compareTo(Student o) {
        return o.name.compareTo(this.name);
    }

在这里插入图片描述

二、Comparator - 比较器

class Student {
    public String name;
    public int age;
    public double score;

    public Student(String name, int age, double score) {
        this.name = name;
        this.age = age;
        this.score = score;
    }

    @Override
    public String toString() {
        return "Student{" +
                "name='" + name + '\'' +
                ", age=" + age +
                ", score=" + score +
                '}';
    }

}

public class TestDome {

    public static void main(String[] args) {
        Student[] students = new Student[3];
        students[0] = new Student("张三", 18, 58.5);
        students[1] = new Student("李四", 10, 90);
        students[2] = new Student("王二", 50, 70.7);
        //排序前
        System.out.println(Arrays.toString(students));
        Arrays.sort(students);
        //排序后
        System.out.println(Arrays.toString(students));
    }
}

最开始的时候比较还是会出错
在这里插入图片描述
进入sort,你会发现有很多比较规则
在这里插入图片描述

现在 我们来创建一个类,该类实现了Comparator接口
Ctrl+左键 进入 Comparator 接口

在这里插入图片描述
输出原理,跟前面讲的,是一样的, 大于就返回一个非零数字,等于返回0,小于返回负数

class Student{
    public String name;
    public int age;
    public double score;

    public Student(String name, int age, double score) {
        this.name = name;
        this.age = age;
        this.score = score;
    }

    @Override
    public String toString() {
        return "Student{" +
                "name='" + name + '\'' +
                ", age=" + age +
                ", score=" + score +
                '}';
    }
}

//根据年龄来比较的比较器
class ageComparator implements Comparator<Student>{
    @Override
    public int compare(Student o1, Student o2) {
        return o1.age - o2.age;
    }
}

那该如何使用比较器呢???

public static void main(String[] args) {
        Student[] students = new Student[3];
        students[0] = new Student("张三", 18, 58.5);
        students[1] = new Student("李四", 10, 90);
        students[2] = new Student("王二", 50, 70.7);
        //排序前
        System.out.println(Arrays.toString(students));
        AgeComparator ageComparator = new AgeComparator();
        Arrays.sort(students,ageComparator);
        //排序后
        System.out.println(Arrays.toString(students));
    }

进入sort方法
在这里插入图片描述
如果要根据分数来比较,那就创建一个根据分数来比较的比较器

class ScoreComparator implements Comparator<Student>{

    @Override
    public int compare(Student o1, Student o2) {
        return (int)(o1.score - o2.score);
    }
}

在这里插入图片描述

根据姓名来排序,那就创建一根据姓名来比较的比较器

class Name implements Comparator<Student>{
    @Override
    public int compare(Student o1, Student o2) {
        return o1.name.compareTo(o2.name);
    }
}

在这里插入图片描述

关于比较器总的代码

class Student{
    public String name;
    public int age;
    public double score;

    public Student(String name, int age, double score) {
        this.name = name;
        this.age = age;
        this.score = score;
    }

    @Override
    public String toString() {
        return "Student{" +
                "name='" + name + '\'' +
                ", age=" + age +
                ", score=" + score +
                '}';
    }

}

class AgeComparator implements Comparator<Student>{

    @Override
    public int compare(Student o1, Student o2) {
        return o1.age - o2.age;
    }
}

class ScoreComparator implements Comparator<Student>{

    @Override
    public int compare(Student o1, Student o2) {
        return (int)(o1.score - o2.score);
    }
}

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

public class TestDome {

    public static void main(String[] args) {
        Student[] students = new Student[3];
        students[0] = new Student("张三", 18, 58.5);
        students[1] = new Student("李四", 10, 90);
        students[2] = new Student("王二", 50, 70.7);
        //排序前
        System.out.println(Arrays.toString(students));

//        ScoreComparator scoreComparator = new ScoreComparator();
//        Arrays.sort(students,scoreComparator);
//        AgeComparator ageComparator = new AgeComparator();
//        Arrays.sort(students,ageComparator);
        NameComparator nameComparator = new NameComparator();
        Arrays.sort(students,nameComparator);
        //排序后
        System.out.println(Arrays.toString(students));
    }
}

三、Cloneable - 克隆

Java 中内置了一些很有用的接口, Clonable 就是其中之一.

Object 类中存在一个 clone 方法, 调用这个方法可以创建一个对象的 “拷贝”.
但是要想合法调用 clone 方法, 必须要先实现 Clonable 接口, 否则就会抛出 CloneNotSupportedException 异常.

class Person{
    public String name;
    public int age;

    public Person(String name, int age) {
        this.name = name;
        this.age = age;
    }

    public void eat(){
        System.out.println(" 正在吃饭!");
    }
    
    @Override
    public String toString() {
        return "Person{" +
                "name='" + name + '\'' +
                ", age=" + age +
                '}';
    }
}

在这里插入图片描述
要想合法调用 clone 方法, 必须要先实现 Clonable 接口, 否则就会抛出 CloneNotSupportedException 异常.

在这里插入图片描述
Cloneable 接口为什么是空的???
一个空接口称为标志接口,表示接口就表示,如果一个类实现了这个和标志接口,那这个类就可以被克隆
这个接口本来就是空的,有些是用语法解释不了的,但是clone方法为什么还是会报错???
那就要在这个类里面重写clone方法
在这里插入图片描述

要克隆一个对象,那这个类就要实现Cloneable接口,并且这个类要重写clone方法

一个对象被克隆,那就产生了一个副本,和这个对象一样

class Person implements Cloneable{
    public String name;
    public int age;

    public Person(String name, int age) {
        this.name = name;
        this.age = age;
    }

    public void eat(){
        System.out.println(" 正在吃饭!");
    }

    @Override
    public String toString() {
        return "Person{" +
                "name='" + name + '\'' +
                ", age=" + age +
                '}';
    }

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

public class Test1 {
    public static void main(String[] args) throws CloneNotSupportedException {
        Person person1 = new Person("张三", 18);
        Person person2 = (Person)person1.clone();
        System.out.println(person2);
    }
}

在这里插入图片描述

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

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

相关文章

MySQL性能优化浅析

1. 硬件 1.1 CPU IO密集型&#xff0c;提升CPU核心数 计算密集型&#xff0c;提升CPU频率 1.2 磁盘 机械硬盘在随机访问时&#xff0c;由于受磁针移动速度的限制&#xff0c;性能会大幅降低。使用固态硬盘可以大幅提升随机访问的能力。按需选择。 1.3 其他 带宽、内存频…

Superset 安装配置

文章目录Superset 安装配置一、Superset 概述1. Superset简介2. 功能概述3. 支持的数据库二、Superset 环境部署步骤三、创建虚拟机&#xff0c;安装CentOS1.下载CentOS2.创建虚拟机3.编辑虚拟机设置4.安装centos7.9mini版本5.启动centos&#xff0c;并进行登录四、CentOS配置1…

小米(Android)刷NetHunter安装指南

一、安装NetHunter 前提&#xff1a;确保手机已经root&#xff0c;已装上magisk。如果没有root&#xff0c;可用尝试magisk root 后执行此文 1、下载Nethunter&#xff1a;Get Kali | Kali Linux 然后push 到sdcard 里&#xff0c; 2、打开magisk&#xff0c;选择刚刚下好的…

Windows下安装libtorch与Clion配置

Windows 安装和使用libtorch 1.下载libtorch libtorch的下载链接&#xff0c;如下图所示&#xff0c;libtorch有release和debug版本可以选择。为了方便调试&#xff0c;下debug版。电脑上没CUDA&#xff0c;下次有需要再更新吧。 2.libtorch使用 在Visual Studio的使用可以参…

云原生周刊 | 让 ChatGPT 以电子邮件的方式来解释 KubeSphere

过去的一周是 ChatGPT 的狂欢&#xff0c;我猜每一位云原生玩家都很好奇他是如何看待 Kubernetes 的。咱们不防换个方式来提问&#xff0c;让它使用电子邮件的方式来向别人推荐 KubeSphere 和 OpenFunction。 开源项目推荐 Tailscale Ingress Controller 这是针对 Tailscale …

【DevOps实战系列】第七章:详解Docker私服Harbor篇

个人亲自录制全套DevOps系列实战教程 &#xff1a;手把手教你玩转DevOps全栈技术 Harbor私服搭建 讲完Nexus3再来看下harbor&#xff0c;其实大同小异&#xff0c;只不过harbor的管理要比Nexus3更专业、功能更完善&#xff0c;大家按需选择即可&#xff0c;Nexus的优势是他能和…

web网站工程项目前期需求分析与规划怎么写?

在当下&#xff0c;判断一份网站工程项目文档是否优秀&#xff0c;项目目录是最直接的体现&#xff0c;同时&#xff0c;工程说明、需求分析和项目规划各版块的内容都缺一不可。工欲善其事必先利其器&#xff0c;前期准备得越充分&#xff0c;后期就会越顺利。 本期&#xff0c…

Centos7安装图形化界面并使用Windows远程桌面连接(包含离线部署)

一、在centos7 中部署远程桌面所使用的程序 1、关闭防火墙和selinux(xrdp是通过3389端口远程桌面连接 ) [rootlocalhost ~]# systemctl stop firewalld #临时关闭防火墙 [rootlocalhost ~]# systemctl disable firewalld.service #永久关闭防火墙 [rootlocalhost ~]# setenf…

SpringMVC:SpringMVC之JSON数据传输参数(5)

JSON数据传输参数1 JSON数据传输参数2 JSON普通数组3 JSON对象数据4 JSON对象数组5 小结1 JSON数据传输参数 现在比较流行的开发方式为异步调用。前后台以异步方式进行交换&#xff0c;传输的数据使用的是JSON,所以前端如果发送的是JSON数据&#xff0c;后端该如何接收? 对于…

面试官:单体架构怎么向分布式微服务架构演变的?(8000字干货)

随着网站规模越来越大&#xff0c;单体应用往往很难再满足要求&#xff0c;就需要向分布式&#xff0c;微服务架构演变。 那么这个演变过程是怎么样的呢&#xff1f;都涉及到哪些组件&#xff0c;会遇到哪些问题&#xff0c;以及相应的解决方案都是什么&#xff0c;本篇文章就…

ESP 低功耗入门

此篇博客以 ESP32 为例来说明 ESP 的睡眠模式。 ESP32 芯片可以大致分成以为五个模块&#xff1a; RF 模块&#xff0c;也就是射频模块&#xff0c;用于蓝牙 / Wi-Fi 的收发(TX / RX)功能。CPU&#xff0c;如 Xtensa 内核 (ESP32 / ESP32-S2 / ESP32-S3)&#xff0c;RISC-V 内…

ssm项目-商城管理系统

1 逆向工程设计 1.1 xml文件配置&#xff08;generatorConfig.xml&#xff09; <?xml version"1.0" encoding"UTF-8"?> <!DOCTYPE generatorConfigurationPUBLIC "-//mybatis.org//DTD MyBatis Generator Configuration 1.0//EN"&q…

滑动窗口的最大值【滑动窗口问题】

文章目录题目解题思路代码展示题目 给定一个长度为 n 的数组 num 和滑动窗口的大小 size &#xff0c;窗口从最左边滑动到最右边&#xff0c;每次向右边滑动一个位置&#xff0c;找出所有滑动窗口里数值的最大值。 窗口大于数组长度或窗口长度为0的时候&#xff0c;返回空 数…

硬件需知知识 -- 基本元件(电阻)

一、电阻 1.1 贴片电阻 1.1.1 贴片电阻的封装大小是和功率时相关的。 封装大小功率(W)0201120\frac{1}{20}201​0402116\frac{1}{16}161​0603110\frac{1}{10}101​080518\frac{1}{8}81​12060.2518120.5或1201012\frac{1}{2}21​25121或者21.1.2 贴片电阻读数 贴片电阻的读数…

Spring Security 竟然可以同时存在多个过滤器链?

目录Spring Security 中的过滤器多个过滤器链多个过滤器链配置例子http.authorizeRequests() 开头是什么意思&#xff1f;引用Spring Security 中的过滤器 Spring Security 中认证、授权功能的实现机制是通过过滤器来实现的。Spring Security 中一共提供了 32 个过滤器&#x…

Nacos集群版本安装(三)

一、前言 注&#xff1a;我这里的nacos集群安装环境使用的是本地MacOs系统&#xff0c;使用的是同一个IP不同端口&#xff0c;这里就不再讲解具体的目录结构之内的了&#xff0c;下面我们开始演示具体的配置和安装。了解nacos单机版本安装 二、nacos集群安装部署&#xff1a; …

Redis常见面试题(二)

目录 1、Redis和Memcached有什么区别? 2、Redis支持哪些数据类型? 3、Redis支持JSON数据类型吗?为什么? 4、Redis模块系统有什么用? 5、Redis支持对象映射模型吗? 6、Redis默认支持多少个数据库?怎么修改? 7、Redis SET命令可以代替SETNX吗? 8、Redis单个实例最…

【浅学Java】索引的分类、创建、删除以及新特性

索引的创建和设计原则1. 索引的分类1.1 普通索引1.2 唯一性索引1.3 主键索引1.4 单列索引1.5 多列&#xff08;联合&#xff0c;组合&#xff09;索引1.6 全文索引2. 索引的创建2.1 创建表时创建索引1. 隐式创建2. 显式创建3. 全文检索2.2 创建表后创建索引1. alter table 的方…

基于java(ssm)家教管理平台(java毕业设计)

基于java&#xff08;ssm&#xff09;家教管理平台 家教管理&#xff0c;是基于java变成语言&#xff0c;mysql数据库&#xff0c;ssm框架和idea工具开发&#xff0c;本系统分为用户&#xff0c;管理员&#xff0c;教师三个角色&#xff0c;其中用户可以注册&#xff0c;登陆&…

网络工程师之海明校验

海明校验&#xff08;又称汉明码&#xff09; 基本思想 将有效信息按某种规律分成若干组&#xff0c;每组安排一个校验位&#xff0c;做奇偶测试&#xff0c;就能提供多位检错信息&#xff0c;以指出最大可能是哪位出错&#xff0c;从而将其纠正。 特点 它不仅具有检测错误的…