言成言成啊 | Kit Chen's Blog

浅谈接口

2019-09-22

一、接口概述与生活举例

啥是接口?不多bb,上图!

接口就是多个类的公共规范。

接口是一种引用数据类型,最重要的内容就是其中的:抽象方法。

二、接口的基本格式

那么,如何定义一个接口的格式

1
2
3
public interface 接口名称 {
//接口内容
}

备注:如果换成了关键字interface之后,编译生成的字节码文件仍然是:.java –> .class

如果是Java 7,那么接口中可以包含的内容有:

  1. 常量
  2. 抽象方法

如果是Java 8,那么接口中还可以额外包含有:

  1. 默认方法
  2. 静态方法

如果是Java 9,还可以额外包含有:

  1. 私有方法

接口使用步骤

  1. 接口不能直接使用,必须有一个”实现类”来”实现”该接口。
    格式:

    1
    2
    3
    public class 实现类名称 implements 接口名称 {
    //...
    }
  2. 接口的实现类必须覆盖重现(实现)接口中所有的抽象方法
    实现:去掉abstract关键字,加上方法体大括号

  3. 创建实现类对象,进行使用。

三、接口的抽象方法

在任何版本Java中,接口都能定义抽象方法。

格式:

1
public abstract 返回值类型 方法名称(参数列表);

注意事项:

  1. 接口当中的抽象方法,修饰符必须是两个固定的关键字:public abstract
  2. 这两个关键字修饰符,可以选择性地省略。(刚学的话,不推荐省略)
  3. 方法的三要素(方法名、参数列表、返回值),可以随意定义。

如果实现类并没有覆盖重写接口中所有的抽象方法,那么这个实现类自己就必须是抽象类。

MyInterfaceAbstract.java

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
public interface MyInterfaceAbstract {

// 这是一个抽象方法
public abstract void methodAbs1();

// 这也是抽象方法
abstract void methodAbs2();

// 这也是抽象方法
public void methodAbs3();

// 这也是抽象方法
void methodAbs4();

}

MyInterfaceAbstractImpl.java

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
public class MyInterfaceAbstractImpl implements MyInterfaceAbstract {
@Override
public void methodAbs1() {
System.out.println("这是第一个方法!");
}

@Override
public void methodAbs2() {
System.out.println("这是第二个方法!");
}

@Override
public void methodAbs3() {
System.out.println("这是第三个方法!");
}

@Override
public void methodAbs4() {
System.out.println("这是第四个方法!");
}
}

Demo01Interface.java

1
2
3
4
5
6
7
8
9
10
11
12
13
public class Demo01Interface {

public static void main(String[] args) {
// 错误写法!不能直接new接口对象使用。
// MyInterfaceAbstract inter = new MyInterfaceAbstract();

// 创建实现类的对象使用
MyInterfaceAbstractImpl impl = new MyInterfaceAbstractImpl();
impl.methodAbs1();
impl.methodAbs2();
}

}

四、接口的默认方法

从java 8开始,接口里允许默认方法

格式:

1
2
3
public default 返回值类型 方法名称(参数列表){
//方法体
}
  1. 接口的默认方法,可以通过接口实现类对象,直接调用。
  2. 接口的默认方法,也可以被接口实现类进行覆盖重写。

备注:接口当中的默认方法,可以解决接口升级的问题。

MyInterfaceDefault.java

1
2
3
4
5
6
7
8
9
10
11
12
13
public interface MyInterfaceDefault {
//抽象方法
public abstract void methodAbs();

//新添加一个抽象方法,会导致两个实现类报错
//public abstract void methodAbs2();

//为了解决上面的问题,需要将新添加的方法,改成默认方法。public省略不写,也会默认为public
public default void methodDefault() {
//一定要将这个大括号写上
System.out.println("这是新添加的默认方法");
}
}

MyInterfaceDefaultA.java

1
2
3
4
5
6
7
8
9
public class MyInterfaceDefaultA implements MyInterfaceDefault {

@Override
public void methodAbs() {
// TODO Auto-generated method stub
System.out.println("实现了抽象方法,AA");
}

}

MyInterfaceDefaultB.java

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
public class MyInterfaceDefaultB implements MyInterfaceDefault {

@Override
public void methodAbs() {
// TODO Auto-generated method stub
System.out.println("实现了抽象方法,BB");
}

@Override
public void methodDefault() {
// TODO Auto-generated method stub
System.out.println("实现类B覆盖重写了接口的默认方法");
}

}

Demo02Interface.java

1
2
3
4
5
6
7
8
9
10
11
12
13
public static void main(String[] args) {

//创建实现类对象
MyInterfaceDefaultA a=new MyInterfaceDefaultA();
a.methodAbs();//调用抽象方法,实际运行的是右侧实现类

//调用默认方法,如果实现类中没有,会向上寻找
a.methodDefault();//这是新添加的默认方法

MyInterfaceDefaultB b=new MyInterfaceDefaultB();
b.methodAbs();
b.methodDefault();//实现类B覆盖重写了默认方法
}

五、接口的静态方法

从Java 8开始,接口中允许定义静态方法。

格式:

1
2
3
public static 返回值类型 方法名称(参数列表){
//方法体
}

提示:

就是将abstract或者default换成static,然后带上方法体。

注意:不能通过接口实现类的对象来调用接口当中的静态方法。

正确用法:直接通过接口名称调用静态方法

格式:

接口名称.静态方法(参数列表)

MyInterfaceStatic.java

1
2
3
4
5
6
7
public interface MyInterfaceStatic {

//public 可以省略
public static void methodStatic() {
System.out.println("这是接口的静态方法!");
}
}

MyInterfaceStaticImpl.java

1
2
3
public class MyInterfaceStaticImpl implements MyInterfaceStatic {

}

Demo03Interface.java

1
2
3
4
5
6
7
8
9
10
11
12
13
public class Demo03Interface {
public static void main(String[] args) {
// MyInterfaceStaticImpl impl=new MyInterfaceStaticImpl();

//错误写法!
//impl.methodStatic();

//直接通过接口名称调用静态方法
MyInterfaceStatic.methodStatic();


}
}

六、接口的私有方法

问题描述:

我们需要抽取一个公共方法,用来解决两个默认方法之间的重复代码的问题,但是这个共有方法不应该让实现类使用,应该是私有化的。

解决方案:

从Java 9开始,接口里允许定义私有方法

  1. 普通私有方法,解决多个默认方法之间重复代码问题

    格式:

    1
    2
    3
    private 返回值类型 方法名称(参数列表){
    //方法体
    }
  2. 静态私有方法,解决多个静态方法之间的重复代码问题
    格式:

    1
    2
    3
    private static 返回值类型 方法名称(参数列表){
    //方法体
    }

MyInterfacePrivateA.java

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
public interface MyInterfacePrivateA {
public default void methodDefault1() {
System.out.println("这是默认方法1");
// System.out.println("AAA");
// System.out.println("BBB");
// System.out.println("CCC");
//methodCommon();

methodPrivateCommon();
}
public default void methodDefault2() {
System.out.println("这是默认方法2");
// System.out.println("AAA");
// System.out.println("BBB");
// System.out.println("CCC");
//methodCommon();

methodPrivateCommon();
}
//如果就像上面的一样,有重复的咋办?
//像下面这样,写个方法呗?但是...

public default void methodCommon() {
System.out.println("AAA");
System.out.println("BBB");
System.out.println("CCC");
}

//用下面这个就完美解决了
private void methodPrivateCommon() {
System.out.println("AAA");
System.out.println("BBB");
System.out.println("CCC");
}
}

MyInterfacePrivateB.java

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
public interface MyInterfacePrivateB {
public static void methodStatic1() {
System.out.println("这是静态方法1");

methodPrivateCommon();
}
public static void methodStatic2() {
System.out.println("这是静态方法2");

methodPrivateCommon();
}


private static void methodPrivateCommon() {
System.out.println("AAA");
System.out.println("BBB");
System.out.println("CCC");
}
}

MyInterfacePrivateAImpl.java

1
2
3
4
5
6
7
public class MyInterfacePrivateAImpl implements MyInterfacePrivateA {
public void methodAnother() {
//methodCommon();//直接访问接口中的默认方法,但是这个不应该被访问到啊


}
}

七、接口的常量定义以及使用

接口当中,也可以定义”成员变量”

但是这个成员变量,需要public static final三个关键字进行修饰

从效果上看,这就是接口的常量

格式:

1
public static final 数据类型 常量名称 = 数据值;

一旦使用final关键字进行修饰,说明不可改变

  1. 接口当中的常量,可以省略这三个关键字,但即使不写,还是默认为自带这三个关键字
  2. 接口当中的常量,必须进行赋值
  3. 接口当中的常量名称,使用完全大写,并且下划线隔开

MyInterfaceConst.java

1
2
3
4
5
6
7
public interface MyInterfaceConst {

//这其实就是一个常量,一旦赋值,是不可以修改的
public static final int NUM_OF_MY_CLASS =10;


}

Demo05Interface.java

1
2
3
4
5
6
public class Demo05Interface {
//访问接口当中的常量
public static void main(String[] args) {
System.out.println(MyInterfaceConst.NUM_OF_MY_CLASS);
}
}

八、接口内容总结

在Java 9+版本中(带有中括号表示可以省略)

  1. 成员变量其实是常量,格式:

    1
    [public] [static] [final] 数据类型 常量名称 = 数据值;

    注意:

    常量必须进行赋值,而且一旦赋值不能改变。
    常量名称完全大写,用下划线进行分割。
  1. 接口中最重要的就是抽象方法,格式:

    1
    [public] [abstract] 返回值类型 方法名称(参数列表);

    注意:实现类必须覆盖重写接口中所有的抽象方法,除非实现类是抽象类。

  2. 从java 8开始,接口里允许定义默认方法,格式:

    1
    [public] default 返回值类型 方法名称(参数列表) { 方法体 }

    注意:默认方法也可以被覆盖重写。

  3. 从java 8开始,接口里允许定义静态方法,格式:

    1
    [public] static 返回值类型 方法名称(参数列表) { 方法体 }

    注意:应该通过接口名称进行调用,不能通过实现类对象调用接口静态方法

  4. 从java 9开始,接口里允许定义私有方法,格式:

    1
    2
    普通私有方法:private 返回值类型 方法名称(参数列表) { 方法体 }
    静态私有方法:private static 返回值类型 方法名称(参数列表) { 方法体 }

    注意:private的方法只有接口自己才能调用,不能被实现类或者别人使用。

九、使用接口时的注意事项

使用接口的时候,需要注意:

  1. 接口是没有静态代码块或者构造方法的

  2. 一个类的直接父类是唯一的,但是一个类可以同时实现多个接口
    格式:

    1
    2
    3
    public class MyInterfaceImpl implements MyInterfaceA,MyInterfaceB{
    //覆盖重写所有抽象方法
    }
  3. 如果实现类所实现的多个接口当中,存在重复的抽象方法,那么只需要覆盖重写一次即可

  4. 如果实现类没有覆盖重写所有接口当中的所有抽象方法,那么实现类就必须是抽象类

  5. 如果实现类实现的多个接口当中,存在重复的默认方法,那么实现类一定要对冲突的默认方法进行覆盖重写

  6. 一个类,如果直接父类当中的方法,跟接口当中继承的方法产生了冲突,优先用父类当中的方法。

MyInterfaceA.java

1
2
3
4
5
6
7
8
9
public interface MyInterfaceA {
public abstract void methodA();

public abstract void methodAbs();

public default void methodDefault() {
System.out.println("默认方法AAA");
}
}

MyInterfaceB.java

1
2
3
4
5
6
7
8
public interface MyInterfaceB {
public abstract void methodB();
public abstract void methodAbs();

public default void methodDefault() {
System.out.println("默认方法BBB");
}
}

MyInterfaceAbstract.java

1
2
3
4
5
6
7
8
public abstract class MyInterfaceAbstract implements MyInterfaceA,MyInterfaceB {

@Override
public void methodDefault() {
System.out.println("不管抽不抽象,有冲突,都得重写");
}

}

MyInterfacImpl.java

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
public class MyInterfaceImpl implements MyInterfaceA,MyInterfaceB {

@Override
public void methodB() {
// TODO Auto-generated method stub
System.out.println("覆盖重写A接口抽象方法");
}

@Override
public void methodA() {
// TODO Auto-generated method stub
System.out.println("覆盖重写B接口抽象方法");
}

@Override
public void methodAbs() {
// TODO Auto-generated method stub
System.out.println("覆盖重写了AB接口都有的抽象方法");
}

@Override
public void methodDefault() {
// TODO Auto-generated method stub
System.out.println("对多个接口当中的冲突的默认方法进行覆盖重写");
}

}

Demo01Interface.java

1
2
3
public class Demo01Interface {

}

接口之间的多继承

  1. 类与类之间是单继承的。直接父类只能有一个。
  2. 类与接口直接是多实现的。一个类可以实现多个接口,有冲突需要解决。
  3. 接口与接口之间是多继承的。

注意事项:

  1. 多个父接口当中的抽象方法如果重复,没关系
  2. 多个父接口当中的默认方法如果重复,那么子接口必须进行默认方法的覆盖重写,而且带着default关键字。
阅读量