【JAVA】抽象类与接口--下(一)

简介: 【JAVA】抽象类与接口--下

1. 实现多个接口


在Java中,类和类之间是单继承的,一个类只能有一个父类,即Java中不支持多继承,但是一个类可以实现多个接口。


通过类来表示一组动物:


class Animal {
  protected String name;
  public Animal(String name) {
  this.name = name;
  }
}

另外我们再提供一组接口,分别表示“会飞的”,“会跑的”,“会游泳的”:


interface IFlying {
  void fly();
}
interface IRunning {
  void run();
}
interface ISwimming {
  void swim();
}

接下来我们创建几个具体的动物:


猫,是会跑的


class Cat extends Animal implements IRunning {
  public Cat(String name) {
  super(name);
  }
  @Override
  public void run() {
  System.out.println(this.name + "正在用四条腿跑");
  }
}

鱼,是会游的


class Fish extends Animal implements ISwimming {
  public Fish(String name) {
    super(name);
  }
  @Override
  public void swim() {
    System.out.println(this.name + "正在用尾巴游泳");
  }
}

青蛙,既能跑,又能游

class Frog extends Animal implements IRunning, ISwimming {
  public Frog(String name) {
    super(name);
  }
  @Override
  public void run() {
    System.out.println(this.name + "正在往前跳");
  }
  @Override
  public void swim() {
    System.out.println(this.name + "正在蹬腿游泳");
  }
}
class Frog extends Animal implements IRunning, ISwimming {
  public Frog(String name) {
    super(name);
  }
  @Override
  public void run() {
    System.out.println(this.name + "正在往前跳");
  }
  @Override
  public void swim() {
    System.out.println(this.name + "正在蹬腿游泳");
  }
}

13

当一个类实现多个接口时,每个接口中的抽象方法都要实现,否则类必须设置为抽象类。


例如:鸭子,会游、会飞、会跑


class Duck extends Animal implements IRunning, ISwimming, IFlying {
    public Duck(String name) {
        super(name);
    }
    @Override
    public void fly() {
        System.out.println(this.name + "正在用翅膀飞");
    }
    @Override
    public void run() {
        System.out.println(this.name + "正在用两条腿跑");
    }
    @Override
    public void swim() {
        System.out.println(this.name + "正在漂在水上");
    }
}

对上述代码的实现:


public class test {
    public static void main(String[] args) {
        Cat cat=new Cat("小花");
        cat.run();
        Fish fish=new Fish("jiujiu");
        fish.swim();
        Frog frog=new Frog("青蛙王子");
        frog.run();
        frog.swim();
        Duck duck=new Duck("可达鸭");
        duck.run();
        duck.fly();
        duck.swim();
    }
}

🍤 运行结果:


dd0c62798defcd906272cc154880edd0_051ef9191072413a8e1052797d1ebfde.png


上面的代码展示了 Java 面向对象编程中最常见的用法:一个类继承一个父类,同时实现多种接口。


🍩继承表达的含义是 is - a 语义,而接口表达的含义是具有 xxx 特性


🍩有了接口之后,类的使用者就不必关注具体类型,而只关注某个类是否具备某种能力


例如,现在实现一个方法,叫“散步”

public static void walk(IRunning running) {
  System.out.println("我带着伙伴去散步");
  running.run();
}

在这个 walk 方法内部,我们并不关注到底是哪种动物,只要参数是会跑的就行。


2. 接口间的继承


在Java中,类和类之间是单继承的,一个类可以实现多个接口,接口与接口之间可以多继承。即:用接口可以达到多继承的目的。

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


interface IRunning {
  void run();
}
interface ISwimming {
  void swim();
}
// 两栖的动物, 既能跑, 也能游
interface IAmphibious extends IRunning, ISwimming {
}
class Frog implements IAmphibious {
  ...
}

通过接口继承创建一个新的接口 IAmphibious 表示“两栖的”,此时实现接口创建的 Frog 类,就继续要实现 run 方法,也需要实现 swim 方法。


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


3. Clonable接口和深拷贝


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

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


class Animals implements Cloneable {
    private String name;
    @Override
    public Animals clone() {
        Animals o = null;
        try {
            o = (Animals)super.clone();
        } catch (CloneNotSupportedException e) {
            e.printStackTrace();
        }
        return o;
    }
}
public class Test1 {
    public static void main(String[] args) {
        Animals animal = new Animals();
        Animals animal2 = animal.clone();
        System.out.println(animal == animal2);
    }
}

🍤 运行结果:

b8927e7743a8b5c093cac063311bbd4f_ffb19dad067248819b6dfabb45e4a90d.png


Cloneable 拷贝出的对象是一份“浅拷贝”


class Money {
    public double m = 99.99;
}
class Person implements Cloneable{
    public Money money = new Money();
    @Override
    protected Object clone() throws CloneNotSupportedException {
        return super.clone();
    }
}
public class Test1 {
    public static void main(String[] args) throws CloneNotSupportedException {
        Person person1 = new Person();
        Person person2 = (Person) person1.clone();
        System.out.println("通过person2修改前的结果");
        System.out.println(person1.money.m);
        System.out.println(person2.money.m);
        person2.money.m = 13.6;
        System.out.println("通过person2修改后的结果");
        System.out.println(person1.money.m);
        System.out.println(person2.money.m);
    }
}

🍤 运行结果:

adb03c853ea14581ba11762bd59a49a7_49fb33adc2d64b4b9ec5124330f80de3.png


如上代码,我们可以看到,通过clone,我们只是拷贝了Person对象。但是Person对象中的Money对象,并没有拷贝。通过person2这个引用修改了m的值后,person1这个引用访问m的时候,值也发生了改变。这里就是发生了浅拷贝。


相关文章
|
5天前
|
Java
Java——抽象类和接口
抽象类是一种不能被实例化的类,至少包含一个抽象方法(无实现体的方法),常用于定义一组相关类的共同特征,并强制子类实现特定方法。抽象方法不能被 `static` 或 `final` 修饰,且必须被重写。 接口则是一个完全抽象的类,用于规范类的行为。接口使用 `interface` 关键字定义,不能实例化,并且类与接口之间是实现关系。 内部类是在一个类内定义的类,分为成员内部类、静态内部类、局部内部类和匿名内部类。成员内部类可被修饰符修饰,静态内部类只能访问外部类的静态成员,局部内部类定义在方法内,匿名内部类则隐藏了名字,直接通过 `new` 关键字定义并实现接口或继承类。
12 5
Java——抽象类和接口
|
5天前
|
Java
Java——接口的使用实例
Comparable接口用于自定义类的对象比较。通过实现此接口并重写`compareTo`方法,可以定义自定义类型的比较规则。 接下来介绍了Comparator接口,它提供了一种更灵活的比较方式。通过实现Comparator接口并重写`compare`方法,可以根据不同属性定义不同的比较规则。例如,定义一个`BrandComparator`类来比较汽车的品牌。 最后,介绍了Cloneable接口,用于实现对象的克隆。实现该接口并重写`clone`方法后,可以创建对象的浅拷贝或深拷贝。浅拷贝仅复制对象本身,深拷贝则会递归复制所有成员变量。
13 4
Java——接口的使用实例
|
11天前
|
Java 数据库连接 数据库
Java服务提供接口(SPI)的设计与应用剖析
Java SPI提供了一种优雅的服务扩展和动态加载机制,使得Java应用程序可以轻松地扩展功能和替换组件。通过合理的设计与应用,SPI可以大大增强Java应用的灵活性和可扩展性。
43 18
|
9天前
|
Java 开发者
Java的接口详解
Java接口是编程中的一种重要特性,用于定义方法签名而不提供具体实现,作为类之间的契约,使不同类能以统一方式交互。接口使用`interface`关键字定义,可包含方法声明和常量。类通过`implements`关键字实现接口,并可同时实现多个接口,解决多重继承问题。接口中的方法默认为抽象方法,变量默认为`public static final`。Java 8引入了默认方法和静态方法,增强接口功能。接口广泛应用于回调机制和多态性实现,有助于构建更灵活和可维护的代码结构。
|
19天前
|
Java
盘点java8 stream中隐藏的函数式接口
`shigen`是一位坚持更新文章的博客作者,记录成长历程,分享认知见解,留住感动瞬间。本文介绍了函数式接口的概念及其在Java中的应用,包括`Comparator`、`Runnable`、`Callable`等常见接口,并详细讲解了`Function`、`Predicate`、`Consumer`、`Supplier`和`Comparator`等函数式接口的使用方法及应用场景,展示了如何利用这些接口简化代码并提高编程效率。**个人IP:shigen**,与shigen一起,每天进步一点点!
29 0
盘点java8 stream中隐藏的函数式接口
|
20天前
|
Java 编译器 开发者
Java中的Lambda表达式与函数式接口
【8月更文挑战第31天】本文将深入探讨Java 8中引入的Lambda表达式和函数式接口,它们如何改变我们编写代码的方式。通过简化集合操作、事件处理等示例,我们将看到这些特性如何提升代码可读性、减少冗余,并提高开发效率。准备好一起探索这个让Java编程更加简洁强大的新世界吧!
|
23天前
|
Java 开发者
Java 8新特性之Lambda表达式与函数式接口
【7月更文挑战第59天】本文将介绍Java 8中的一个重要新特性——Lambda表达式,以及与之密切相关的函数式接口。通过对比传统的匿名内部类,我们将探讨Lambda表达式的语法、使用方法和优势。同时,我们还将了解函数式接口的定义和用途,以及如何将Lambda表达式应用于函数式编程。
|
22天前
|
Java
在Java多线程领域,精通Lock接口是成为高手的关键。
在Java多线程领域,精通Lock接口是成为高手的关键。相较于传统的`synchronized`,Lock接口自Java 5.0起提供了更灵活的线程同步机制,包括可中断等待、超时等待及公平锁选择等高级功能。本文通过实战演练介绍Lock接口的核心实现——ReentrantLock,并演示如何使用Condition进行精确线程控制,帮助你掌握这一武林秘籍,成为Java多线程领域的盟主。示例代码展示了ReentrantLock的基本用法及Condition在生产者-消费者模式中的应用,助你提升程序效率和稳定性。
18 2
|
22天前
|
Java 开发者
在 Java 多线程编程中,Lock 接口正逐渐取代传统的 `synchronized` 关键字,成为高手们的首选
在 Java 多线程编程中,Lock 接口正逐渐取代传统的 `synchronized` 关键字,成为高手们的首选。相比 `synchronized`,Lock 提供了更灵活强大的线程同步机制,包括可中断等待、超时等待、重入锁及读写锁等高级特性,极大提升了多线程应用的性能和可靠性。通过示例对比,可以看出 Lock 接口通过 `lock()` 和 `unlock()` 明确管理锁的获取和释放,避免死锁风险,并支持公平锁选择和条件变量,使其在高并发场景下更具优势。掌握 Lock 接口将助力开发者构建更高效、可靠的多线程应用。
18 2
|
23天前
|
Java 开发者
Java中的接口和抽象类
Java中的接口和抽象类
24 3