摘要

啥是接口?你看了就懂!

正文

一、接口概述与生活举例

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

接口.png

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

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

二、接口的基本格式

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

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

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

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

  1. 常量
  2. 抽象方法

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

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

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

  1. 私有方法

接口使用步骤

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

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

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

三、接口的抽象方法

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

格式:

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

注意事项:

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

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

MyInterfaceAbstract.java

text
 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

text
 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

text
 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开始,接口里允许默认方法

格式:

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

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

MyInterfaceDefault.java

text
 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

text
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

text
 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

text
 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开始,接口中允许定义静态方法。

格式:

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

提示:

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

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

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

格式:

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

MyInterfaceStatic.java

text
1
2
3
4
5
6
7
public interface MyInterfaceStatic {
	
	//public 可以省略
	public static void methodStatic() {
		System.out.println("这是接口的静态方法!");
	}
}

MyInterfaceStaticImpl.java

text
1
2
3
public class MyInterfaceStaticImpl implements MyInterfaceStatic {

}

Demo03Interface.java

text
 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. 普通私有方法,解决多个默认方法之间重复代码问题

    格式:

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

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

MyInterfacePrivateA.java

text
 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

text
 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

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

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

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

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

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

格式:

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

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

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

MyInterfaceConst.java

text
1
2
3
4
5
6
7
public interface MyInterfaceConst {
	
	//这其实就是一个常量,一旦赋值,是不可以修改的
	public static final int NUM_OF_MY_CLASS =10;
	
	
}

Demo05Interface.java

text
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. 成员变量其实是常量,格式:

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

    注意: 常量必须进行赋值,而且一旦赋值不能改变。 常量名称完全大写,用下划线进行分割。

  2. 接口中最重要的就是抽象方法,格式:

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

MyInterfaceA.java

text
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

text
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

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

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

MyInterfacImpl.java

text
 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

text
1
2
3
public class Demo01Interface {

}

接口之间的多继承

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

注意事项:

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