重新总结 Java 中的接口

简介: 重新总结 Java 中的接口

在之前的博客中,我有写到接口,但是还有一些语法没有总结到。本篇博客旨在于将接口拎出来单独总结。


一、接口



接口从某种意义上来说,就是完完全全的抽象类。


接口:由 interface 实现,如:


interface IA {
}


1. 接口的语法规则


① 接口中的普通方法,不能有具体实现,如果非要实现,需要被 default 修饰这个方法

② 接口中的所有方法的限定符都是 public,那么所有的方法都可以省略 public,而接口中的所有抽象方法的限定符都是 abstract public,那么所有的抽象方法同样可以省略 abstract public

③ 接口中的成员变量默认是被 public static final 修饰的,也就是说,接口中的成员变量必须被初始化

接口不可以通过 new 实例化对象

⑤ 如果一个类实现了一个接口,那么在这个类中,必须重写接口中所有的抽象方法,而且重写的抽象方法必须被 public 修饰(因为接口中的抽象方法默认是 public abstract)

⑥ 如果一个类 A 实现了接口 B,那么对应的代码格式为:


interface IB{
}
class A implements IB{
}


⑦ 一个类可以实现多个接口,然而一个类只能继承一个类


interface IX{
}
interface IY{
}
class Z{
}
class A extends Z implements IX,IY{
}


⑧ 两个接口之间使用 extends,表示一个接口拓展另一个接口。

注意下面代码,当类 A 实现了接口 IY ,类 A 必须重写两个方法。


interface IX{
    void func1();
}
interface IY extends IX{
    void func2();
}
class A implements IY{
    @Override
    public void func2() {
    }
    @Override
    public void func1() {
    }
}


2. 通过代码来演示接口的语法


在程序清单1中,我通过注释标明了接口使用时的一些语法规则,这十分重要!


程序清单1:

interface IShape{
    public int a; //error
    public static final int b = 10; //right
    int c = 20; //right
    public void func1(){ //error
    }
    default public void func2(){ //right
    }
    public static void write(){ //right
    }
    abstract public void draw1(){ //error
    }
    abstract public void draw2(); //right
    void draw3(); //right
}
public class Test {
    public static void main(String[] args) {
        IShape iShape = new IShape(); //error
    }
}


3. 继承与接口的综合


程序清单2:


class Animal{
    String name;
    public Animal(String name) {
        this.name = name;
    }
    public void eat(){
        System.out.println(name + " 正在吃东西");
    }
}
interface IFlying{
    void fly();
}
class Bird extends Animal implements IFlying{
    public Bird(String name) {
        super(name);
    }
  //重写了接口中的方法
    @Override
    public void fly() {
        System.out.println(name + " 正在飞");
    }
  //重写了父类中的方法
    @Override
    public void eat() {
        System.out.println(name + " 正在喝蜂蜜");
    }
}
public class Test {
    public static void fly(IFlying iFlying){
        iFlying.fly();
    }
    public static void main(String[] args) {
        fly(new Bird("老鹰"));
        new Bird("蜜蜂").eat();
    }
}


输出结果:


2bbf3eb01da44655b1bbef3560e419ab.png


对程序清单2 进行分析:


在程序清单2 中,我们演示了继承和接口,有几个语法点很关键:


① 在父类构造带参数的方法的同时,子类也需要构造同样的带参数方法。


public Bird(String name) {
}


② 在父类已有的普通成员方法情况下,子类却重写了父类的此方法,那么在编译时,系统直接使用子类重写的方法。


public void eat(){
}


③ 接口中的抽象方法是被 abstract public 修饰的,那么在一个类实现接口的时候,我们就必须对此抽象方法进行重写。


@Override
public void fly() {
  System.out.println(name + " 正在飞");
}


二、为什么有些代码 new 了一个接口



演示1


通常情况下,一个类实现了另一个接口,我们只能使用 implements 显示地表现出来,如下面的程序:


程序清单3:

interface IA{
    void test();
}
class B implements IA {
    @Override
    public void test() {
        System.out.println("我通过类 B 重写了接口的 test 方法");
    }
}
public class Test {
    public static void main(String[] args) {
        B b = new B();
        b.test();
    }
}


输出结果:


8c479d5d902144119d268e146c0d2c1b.png


演示2


而我们知道接口是不能通过 new 来实例化对象的,但有时候我们也会看到【 new 接口 】的情况出现,但不会被编译器报错。因为这是一个匿名内部类实现接口的缘故,只是这个匿名类看起来被隐藏了一样,给人一种 new 了一个接口的错觉,而实际上匿名内部类实现了另一个接口,那么它依然需要重写接口中的抽象方法。所以,一个接口依然不能通过 new 实例化对象!


//语法格式
new + 接口名 + {
  //这里必须重写接口中的抽象方法!
};


通过下列程序来进行演示为什么可以这么做:


程序清单4:


interface IA{
    void test();
}
public class Test {
    public static void main(String[] args) {
        IA ia = new IA(){ //1
            @Override
            public void test() { //2
                System.out.println("我匿名重写了接口的 test 方法");
            }
        }; 
        ia.test();//3
    }
}
//注释
//1. 匿名类实现了接口 IA
//2. 匿名类重写了 test 方法
//3. 匿名类调用了 test 方法


输出结果:


f44a3d1ae7184675a0dd8808774c25df.png


演示3


程序清单4 又可以简化成程序清单5


程序清单5:


interface IA{
    void test();
}
public class Test1 {
    public static void main(String[] args) {
        new IA(){ //1
            @Override
            public void test() { //2
                System.out.println("我匿名重写了接口的 test 方法");
            }
        }.test(); //3
    }
}
//注释
//1. 匿名类实现了接口 IA
//2. 匿名类重写了 test 方法
//3. 匿名类调用了 test 方法


输出结果:


4e9e56a8da434c3e98b062e26f8409c5.png


演示4


而程序清单5 又可以写成 Lambda 表达式


( (IA) () -> { System.out.println("我匿名重写了接口的 test 方法");} ).test();


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