【Java SE】抽象类和接口(下)

简介: 首先来看到抽象这两个字,抽象其实是与具体对应的概念,在我们面向对象的概念中,所有的对象都是由类来描述的,如果我们反过来呢?

2.5 实现多个接口

在类的学习中,我们学到Java中的类是不能多继承的,但是一个类可以实现多个接口,就拿我们之前的动物类来说,我有一个猫,他是动物,他既能跑但是不会游泳,但是青蛙,也是动物,他既能跑又会游泳,我们该如何用代码表示呢?

interface IRun {
    void run();
}
interface ISwimming {
    void swimming();
}
public class Animal {
    private String name;
    public Animal() {} //构造方法
    public Animal(String name) {
        this.name = name;
    }
    public String getName() {
        return name;
    }
}
class Cat extends Animal implements IRun {
    public Cat() {} //构造方法
    public Cat(String name) {
        super(name);
    }
    @Override
    public void run() {
        System.out.println(super.getName() + "正在跑!");
    }
}
class Frog extends Animal implements IRun, ISwimming {
    public Frog() {} //构造方法
    public Frog(String name) {
        super(name);
    }
    @Override
    public void run() {
        System.out.println(super.getName() + "正在跑!");
    }
    @Override
    public void swimming() {
        System.out.println(super.getName() + "正在游泳!");
    }
}
class Test {
    public static void running(IRun r) {
        r.run();
    }
    public static void swimming(ISwimming s) {
        s.swimming();
    }
    public static void main(String[] args) {
        new Cat("小猫").run();
        Frog frog = new Frog("青蛙");
        frog.run();
        frog.swimming();
    }
}

如上代码就是Java面向对象最常见的:一个类继承了一个类实现了多种接口,当然这里我们只是很简单的演示一下,像上面也就是说,青蛙既能跑又能游泳,所以实现了跑和游泳的接口,接着我们自己在外部实现一个对应行为的方法比如 running 方法,这样一来,我们就不去关心类型,只用关心某个类能否具备某种能力(是否实现了对应接口),甚至你添加一个机器人类,实现跑的接口,仍然可以放到 running 这个方法中一样能跑,因为我们只关心机器人具不具备跑的能力,而不去关心他机器人的类型是什么!

2.6 接口之间的继承

学习类我们之类,Java中的类不支持多继承,但是一个类可以实现多个接口,而接口与接口之间可以达到多继承的目的。

接口可以继承另一个接口,达到复用的效果,使用 extends 关键字:

interface IRunning {
    void run();
}
interface ISwimming {
    void swimming();
}
//两栖动物,既能跑也能游泳,还能吃饭
interface IAmphibious extends IRunning, ISwimming {
    void eat();
}
class Frog implements IAmphibious {
    //...code 需要实现IRunning,ISwimming,IAmphibious方法
}

接口之间的继承相当于把多个接口合并在一起。

2.7 抽象类和接口的区别

这个也是面试可能会问到一道题,那么抽象类和接口有什么区别呢?

从结构组成来看:抽象类是由普通类加抽象方法构成的,而接口是由抽象方法加全局变量构成的。

从关系上来看:一个抽象类可以实现若干个接口,接口不能继承抽象类, 但可以使用 extends 继承多个父类接口

从子类限制来看:一个子类只能继承一个抽象类,一个子类可以实现多个接口。

核心区别:抽象类中可以包含普通方法和普通字段,这样的普通方法和字段可以被子类直接使用(不用重写),而接口中不能包含普通方法(default,static修饰除外),子类必须重写所有的抽象方法。抽象类的存在是为了让编译器更好的校验!

3、Object 类

3.1 认识Object类

这个类其实每个人都用过,但是对于初学者可能没发现,Object 是Java默认提供的一个类,Java中除了 Object 类,所有的类都是存在继承关系,也就是说,都会默认继承 Object 类,即所有对象的都可以使用 Object 的引用来接收,例如:

class Student {}
class Teacher {}
public class TestDemo {
    public static void function(Object o) {
        System.out.println(o);
    }
    public static void main(String[] args) {
        function(new Student());
        function(new Teacher());
    }
}

打印结果:

Student@1b6d3586

Teacher@4554617c

当然Objcet类中里面也有许多方法,后期我们会随着学习的深入,都会学习到,而本期我们就简单的了解两个方法:toString,equals 方法

3.2 获取对象信息 toString

不知道在打印数组的时候我们使用 Arrays.toString(array),就可以将array数组转换成字符串返回,我们来简单看下 Object 类中 toString 的实现方法:

这里小伙伴就纳闷了,这行代码为何能将数组转换成字符串呢?错了错了!看清楚,是Arrays类里面的toString是将数组转换成字符串返回的,为什么 Arrays 类里面的可以实现?因为除 Object 类之外所有类都默认继承了 Object 类,也就是可以重写 Object 类中的方法, 也即 Arrays 类中重写的 toString 方法:

所以如果我们要将对象信息转换成字符串,在没有重写 toString 方法的前提下,也就会打印像我们上面举例的代码一样,打印出:Student@1b6d3586 类似这种东西,那我们就来重写下 toString:

class Student {
    private String name;
    private int age;
    private 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 TestDemo {
    public static void main(String[] args) {
        Student student = new Student("李四", 15, 83);
        String ret = student.toString();
        System.out.println(ret);
        System.out.println(new Student("张三", 12, 59.9).toString());
    }
}

当然这个重写是我自己用编译器生成的,如果你是 IDEA 可以在要重写的类中右击选择 Generat -> toString(),可以根据情况选择内容,然后确认即可,当然生成之后你也可以做修改。

上面代码第一种打印方法,是把 toString 的返回的字符串给了一个字符串引用,在通过打印方法进行打印,而第二种则是 new 了一个匿名对象直接访问重写的 toString方法,然后配合打印方法进行打印,当然匿名对象会在后续学习。

3.3 对象的比较 equals

在Java中使用 == 号比较的话,如果左右两侧是基本类型,比较的是变量中的值是否相同,但如果左右两边是引用类型,则比较引用变量的地址是否相同,因为引用变量中存的就是地址!

显然我们要比较对象的大小肯定是要重写的,我们可以先来看下 Object 类中的 equals 方法:

那我们要比较对象的大小,对象中又有这么多的成员变量,肯定是需要挑一个属性作为比较的,所以我们就选择 age 作为比较:

class Student {
    private String name;
    private int age;
    private double score;
    public Student(String name, int age, double score) {
        this.name = name;
        this.age = age;
        this.score = score;
    }
    @Override
    public boolean equals(Object o) {
        if (o == null) {
            return false;
        }
        if (this == o) {
            return true; //当前对象引用的地址是否等于要比较对象的地址
        }
        //如果不是Student类对象
        if (!(o instanceof Student)) {
            return false;
        }
        //使用age作比较 注意:因为o是Object类型,向下转型才能访问Student类的属性
        return (this.age == ((Student) o).age);
    }
}
public class TestDemo {
    public static void main(String[] args) {
        Student student1 = new Student("李四", 15, 83);
        Student student2 = new Student("张三", 20, 59.9);
        if (student1.equals(student2)) {
            System.out.println("student1 > student2");
        } else {
            System.out.println("student1 < student2");
        }
    }
}

那么我们重写的 equals 就是按照年龄进行比较的, 如果 student1的 age 等于 student2的 age 我们就返回 true 否则返回 false,当然我们也可以按照姓名来比较,使用 this.name.equals 这个 equals就行,因为String类型本身就是引用类型,他里面也重写了 equals,至于如何操作,就交给你们啦!

相关文章
|
4月前
|
消息中间件 缓存 前端开发
从资损百万到零事故:Java 接口幂等设计的艺术与实践
在分布式系统中,重复请求常引发严重资损,如支付双扣、库存超卖等问题,其根源在于接口缺乏幂等性设计。本文通过真实案例揭示幂等性的重要性,并详解8种主流解决方案,涵盖唯一请求ID、乐观锁、悲观锁、状态机等,帮助开发者构建稳定系统,保障业务一致性。无论你是架构师还是开发工程师,都能从中获得实战指导,有效规避重复调用带来的风险。
347 2
|
4月前
|
数据采集 JSON Java
Java爬虫获取1688店铺所有商品接口数据实战指南
本文介绍如何使用Java爬虫技术高效获取1688店铺商品信息,涵盖环境搭建、API调用、签名生成及数据抓取全流程,并附完整代码示例,助力市场分析与选品决策。
|
3月前
|
算法 安全 Java
除了类,Java中的接口和方法也可以使用泛型吗?
除了类,Java中的接口和方法也可以使用泛型吗?
162 11
|
2月前
|
Java Go 开发工具
【Java】(9)抽象类、接口、内部的运用与作用分析,枚举类型的使用
抽象类必须使用abstract修饰符来修饰,抽象方法也必须使用abstract修饰符来修饰,抽象方法不能有方法体。抽象类不能被实例化,无法使用new关键字来调用抽象类的构造器创建抽象类的实例。抽象类可以包含成员变量、方法(普通方法和抽象方法都可以)、构造器、初始化块、内部类(接 口、枚举)5种成分。抽象类的构造器不能用于创建实例,主要是用于被其子类调用。抽象类中不一定包含抽象方法,但是有抽象方法的类必定是抽象类abstract static不能同时修饰一个方法。
232 0
|
4月前
|
存储 缓存 安全
Java集合框架(二):Set接口与哈希表原理
本文深入解析Java中Set集合的工作原理及其实现机制,涵盖HashSet、LinkedHashSet和TreeSet三大实现类。从Set接口的特性出发,对比List理解去重机制,并详解哈希表原理、hashCode与equals方法的作用。进一步剖析HashSet的底层HashMap实现、LinkedHashSet的双向链表维护顺序特性,以及TreeSet基于红黑树的排序功能。文章还包含性能对比、自定义对象去重、集合运算实战和线程安全方案,帮助读者全面掌握Set的应用与选择策略。
295 23
|
3月前
|
安全 Java API
Java SE 与 Java EE 区别解析及应用场景对比
在Java编程世界中,Java SE(Java Standard Edition)和Java EE(Java Enterprise Edition)是两个重要的平台版本,它们各自有着独特的定位和应用场景。理解它们之间的差异,对于开发者选择合适的技术栈进行项目开发至关重要。
466 1
|
4月前
|
安全 Java 开发者
Java集合框架:详解Deque接口的栈操作方法全集
理解和掌握这些方法对于实现像浏览器后退功能这样的栈操作来说至关重要,它们能够帮助开发者编写既高效又稳定的应用程序。此外,在多线程环境中想保证线程安全,可以考虑使用ConcurrentLinkedDeque,它是Deque的线程安全版本,尽管它并未直接实现栈操作的方法,但是Deque的接口方法可以相对应地使用。
273 12
|
4月前
|
存储 安全 Java
Java集合框架(一):List接口及其实现类剖析
本文深入解析Java中List集合的实现原理,涵盖ArrayList的动态数组机制、LinkedList的链表结构、Vector与Stack的线程安全性及其不推荐使用的原因,对比了不同实现的性能与适用场景,帮助开发者根据实际需求选择合适的List实现。
|
4月前
|
Java API 网络架构
java调用api接口自动判断节假日信息
java调用api接口自动判断节假日信息
1448 0
|
Java
Java接口和抽象类
Java接口和抽象类
207 0