Java——接口的使用实例

简介: Comparable接口用于自定义类的对象比较。通过实现此接口并重写`compareTo`方法,可以定义自定义类型的比较规则。接下来介绍了Comparator接口,它提供了一种更灵活的比较方式。通过实现Comparator接口并重写`compare`方法,可以根据不同属性定义不同的比较规则。例如,定义一个`BrandComparator`类来比较汽车的品牌。最后,介绍了Cloneable接口,用于实现对象的克隆。实现该接口并重写`clone`方法后,可以创建对象的浅拷贝或深拷贝。浅拷贝仅复制对象本身,深拷贝则会递归复制所有成员变量。

Comparable接口

当我们想要比较两个基本数据类型的大小时直接用 > , < , = 就可以了,那么如果是自定义的类要根据什么规则进行比较呢?这就用到了Comparable接口,接口中定义的就是一种规范,通过重写接口中的compareTo方法,定义比较规则,就实现了自定义类型的比较

//调用接口
class Man implements Comparable<Man>{
    public String name;
    public int age;
    public Man(String name,int age){
        this.name = name;
        this.age = age;
    }
    
    @Override
    public String toString() {
        return "Student{" +
                "name='" + name + '\'' +
                ", age=" + age +
                '}';
    }
//重写接口方法
    @Override
    public int compareTo(Man m) {
        return this.age - m.age;
    }
}
public class Text01 {
    public static void main(String[] args) {
        Man man1 = new Man("li",18);
        Man man2 = new Man("wang",20);
        System.out.println(man1.compareTo(man2));//输出-2
    }
}

谁调用compareTo方法,this就代表谁

String类型比较

上面比较的age是整型,可以相减,那如果是String 类型呢

可以看出,Java中的String类也实现了Comparable接口,同时也重写了compareTo方法

那么我们只需要调用就可以了

@Override
    public int compareTo(Man m) {
        return this.name.compareTo(m.name);
    }

多个对象的比较

import java.util.Arrays;
class Man implements Comparable<Man>{
    public String name;
    public int age;
    public Man(String name,int age){
        this.name = name;
        this.age = age;
    }
    @Override
    public String toString() {
        return "Student{" +
                "name='" + name + '\'' +
                ", age=" + age +
                '}';
    }
    @Override
    public int compareTo(Man m) {
        return this.name.compareTo(m.name);
    }
}
public class Text01 {
    public static void main(String[] args) {
        Man man1 = new Man("li",18);
        Man man2 = new Man("wang",20);
        Man man3 = new Man("zhang",22);
        Man[] men = {man1,man2,man3};
        //使用Arrays类的sort方法对数组进行排序
        Arrays.sort(men);
        System.out.println(Arrays.toString(men));
    }
}

运行结果:

可以看出,是按照重写后的compareTo方法进行排序的


这一次程序报错了,类型转换的错误

当点到出错的地方来看

当把之前的注释取消,源码底层就会把Man强转为Comparable,调用compareTo方法,而此时由于compareTo重写了,就会调用重写后的方法,排序也会根据重写之后的compareTo方法进行排序。

简单模拟一下Array.sort对自定义类型的排序

public static void main(String[] args) {
        Man man1 = new Man("li",18);
        Man man2 = new Man("wang",20);
        Man man3 = new Man("zhang",22);
        Man[] men = {man1,man2,man3};
        //使用Mysort方法对数组进行排序
        Mysort(men);
        System.out.println(Arrays.toString(men));
    }
public static void Mysort(Comparable[] comparables){
        for(int i = 0;i < comparables.length - 1;i++){
            for(int j = 0;j< comparables.length - 1 - i;j++){
                //如果后面的元素比前面的元素小,则交换它们的位置
                if(comparables[j].compareTo(comparables[j+1]) > 0){
                    Comparable temp = comparables[j];
                    comparables[j] = comparables[j+1];
                    comparables[j+1] = temp;
                }
            }
        }
    }

所以当调用Arrays.sort给自定义的类型进行排序的时候,也必须实现Comparable的接口

但是无论是比较age还是name,用以上方法都比较固定,不够灵活,为了解决这个问题,我们来介绍下一个接口

Comparator接口

public class BrandComparator implements Comparator<Car>{
    @Override
    public int compare(Car o1, Car o2) {
        return o1.getBrand().compareTo(o2.getBrand());
    }
}

调用接口之后重写compare方法

public static void main(String[] args){
        Car car1 = new Car("benz","black",200);
        Car car2 = new Car("audi","white",250);
        Car[] cars = {car1,car2};
        //使用Comparator接口的compare方法
        BrandComparator brandComparator = new BrandComparator();
        //传入比较器
        Arrays.sort(cars,brandComparator);
        //toString方法
        System.out.println(Arrays.toString(cars));
        for(Car car:cars){
            System.out.println("品牌:" + car.getBrand());
            System.out.println("颜色:" + car.getColor());
            System.out.println("速度:" + car.getSpeed());
        }
    }

也可以直接调用方法,单独比较

BrandComparator brandComparator = new BrandComparator();
        int res = brandComparator.compare(car1,car2);
        System.out.println(res);

这样就比较灵活,想要根据哪个属性比较,就定义一个类去实现Comparator接口,再定义比较的规则

Cloneable接口

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

创建一个Person类之后,实现对象的克隆

虽然Person类里面没有clone方法,但是Object类里面有,每一个类都默认继承与Object类,但此时还是报错了,这就需要在Person类中重写clone方法

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

加上之后还是错的,原因是,自定义类型想要进行克隆还要实现Cloneable接口

点进源码发现Cloneable接口没有写任何方法,是一个空接口,空接口也叫做标记接口,表示当前类是可以被克隆的

之后还要处理一下异常,再对返回值类型进行强转,因为此时返回的是Object类型的,这样就可以正常执行了

public static void main(String[] args) throws CloneNotSupportedException{
        Person person1 = new Person("张三", 20);
        Person person2 = (Person) person1.clone();
    }


具体过程就是创建person1对象之后,再克隆一份数据,创建person2,把克隆出的数据赋值给person2

浅拷贝


在原来的基础上再加上一个Money类,此时再对克隆后的值进行修改


可以看出,当把person1的money修改之后,两个是都会发生改变的,这种就叫做浅拷贝

此时就是只克隆了Person的对象,没有克隆Money的对象

深拷贝

如果想要Money也进行克隆,就需要实现Cloneable接口,同时重写clone方法

class Money implements Cloneable{
    public double money = 9.9;
    @Override
    protected Object clone() throws CloneNotSupportedException {
        return super.clone();
    }
}

Person类的clone也要重写


Person克隆之后,Money也进行克隆,就是深拷贝

相关文章
|
9天前
|
JSON Java Apache
非常实用的Http应用框架,杜绝Java Http 接口对接繁琐编程
UniHttp 是一个声明式的 HTTP 接口对接框架,帮助开发者快速对接第三方 HTTP 接口。通过 @HttpApi 注解定义接口,使用 @GetHttpInterface 和 @PostHttpInterface 等注解配置请求方法和参数。支持自定义代理逻辑、全局请求参数、错误处理和连接池配置,提高代码的内聚性和可读性。
|
10天前
|
Java
java线程接口
Thread的构造方法创建对象的时候传入了Runnable接口的对象 ,Runnable接口对象重写run方法相当于指定线程任务,创建线程的时候绑定了该线程对象要干的任务。 Runnable的对象称之为:线程任务对象 不是线程对象 必须要交给Thread线程对象。 通过Thread的构造方法, 就可以把任务对象Runnable,绑定到Thread对象中, 将来执行start方法,就会自动执行Runable实现类对象中的run里面的内容。
24 1
|
15天前
|
Java 开发者
在Java多线程编程的世界里,Lock接口正逐渐成为高手们的首选,取代了传统的synchronized关键字
在Java多线程编程的世界里,Lock接口正逐渐成为高手们的首选,取代了传统的synchronized关键字
41 4
|
21天前
|
安全 Java
在 Java 中使用实现 Runnable 接口的方式创建线程
【10月更文挑战第22天】通过以上内容的介绍,相信你已经对在 Java 中如何使用实现 Runnable 接口的方式创建线程有了更深入的了解。在实际应用中,需要根据具体的需求和场景,合理选择线程创建方式,并注意线程安全、同步、通信等相关问题,以确保程序的正确性和稳定性。
|
20天前
|
Java
Java基础(13)抽象类、接口
本文介绍了Java面向对象编程中的抽象类和接口两个核心概念。抽象类不能被实例化,通常用于定义子类的通用方法和属性;接口则是完全抽象的类,允许声明一组方法但不实现它们。文章通过代码示例详细解析了抽象类和接口的定义及实现,并讨论了它们的区别和使用场景。
|
20天前
|
Java 测试技术 API
Java零基础-接口详解
【10月更文挑战第19天】Java零基础教学篇,手把手实践教学!
18 1
|
25天前
|
Java 开发者
在Java多线程编程中,创建线程的方法有两种:继承Thread类和实现Runnable接口
【10月更文挑战第20天】在Java多线程编程中,创建线程的方法有两种:继承Thread类和实现Runnable接口。本文揭示了这两种方式的微妙差异和潜在陷阱,帮助你更好地理解和选择适合项目需求的线程创建方式。
19 3
|
25天前
|
Java
通过Java代码解释成员变量(实例变量)和局部变量的区别
本文通过一个Java示例,详细解释了成员变量(实例变量)和局部变量的区别。成员变量属于类的一部分,每个对象有独立的副本;局部变量则在方法或代码块内部声明,作用范围仅限于此。示例代码展示了如何在类中声明和使用这两种变量。
|
25天前
|
Java
在Java多线程编程中,实现Runnable接口通常优于继承Thread类
【10月更文挑战第20天】在Java多线程编程中,实现Runnable接口通常优于继承Thread类。原因包括:1) Java只支持单继承,实现接口不受此限制;2) Runnable接口便于代码复用和线程池管理;3) 分离任务与线程,提高灵活性。因此,实现Runnable接口是更佳选择。
33 2
|
25天前
|
Java
Java中多线程编程的基本概念和创建线程的两种主要方式:继承Thread类和实现Runnable接口
【10月更文挑战第20天】《JAVA多线程深度解析:线程的创建之路》介绍了Java中多线程编程的基本概念和创建线程的两种主要方式:继承Thread类和实现Runnable接口。文章详细讲解了每种方式的实现方法、优缺点及适用场景,帮助读者更好地理解和掌握多线程编程技术,为复杂任务的高效处理奠定基础。
28 2