接口场景
思考:在Java中只支持单继承,那么如何解决一个类型中需要兼容多种类型特征的问题?以及多个不同类型具有相同特征的问题呢?
问题:描述手机的发展史
比如原始的手机只有打电话的功能,随着技术的不断发展,手机的功能逐渐变得强大,可以发短信、看视频、听音乐、拍照上网玩游戏了等等,面对这样的场景描述,如何进行合理的代码实现?
代码实现:利用继承的特性进行实现
相机也可以拍照,电脑也可以上网,智能手表也可以打电话、发短信,那么如果要定义这几个类,我们显然无法抽取出公共的父类,只能每种类型都定义一个类去描述它们的功能。
这样虽然可以实现,但是它们之间是不是真的没办法去建立关联呢,虽然它们无法抽取出公共的父类,但是它们当中很多类型之间是具有相同行为能力的,比如发短信,那么是否可以根据这种行为能力去对它们建立联系呢?在java当中就可以通过定义接口实现行为的关联。
接口定义
我们将拍照功能定义为一个接口
/**
* 具有照相能力的接口
*/
public interface IPhoto {
//拍照方法
public void photo();
}
然后分别让手机类和相机类去实现这个接口
//第四代手机
public class FourthPhone implements IPhoto{
@Override
public void photo(){
System.out.println();
System.out.println("手机可以拍照");
}
public void network(){
System.out.println("手机可以上网");
}
public void game(){
System.out.println("手机可以玩游戏");
}
}
//相机
public class Camera implements IPhoto {
@Override
public void photo() {
System.out.println("相机可以拍照");
}
}
和继承一样,依然可以利用多态性质让接口引用指向实现类实例
IPhoto photo = new FourthPhone();
photo.photo(); //手机可以拍照
IPhoto photo2 = new Camera();
photo2.photo(); //相机可以拍照
总结:可以通过接口来去描述不同的类型具有相似的行为特征,建立关系之后以接口引用指向实现类的方式来去描述不同的类型对于接口行为的具体表现。
接口
接口定义了某一批类所需要遵守的规范。
接口不关心这些类的内部数据,也不关心这些类里方法的实现细节,它只规定这些类里必须提供某些方法。
语法
[修饰符] interface 接口名 [extends 父接口1,父接口2...] { 常量定义,抽象方法定义 }
- 接口中可以定义常量和抽象方法(1.8以前),接口的修饰符可以为public、默认
- 接口方法默认修饰符为
public abstract
,可以省略不写 - 常量默认修饰符为
public static final
,可以省略不写
注意事项
- 当类实现接口时,需要去实现接口中的所有抽象方法,否则需要将该类设置为抽象类
- 一般接口的名称以大写 I 开头,容易区别于类
- 接口中没有构造方法,不能被实例化
- 接口不能使用private修饰符
jdk1.8新特性
- jdk1.8中可以在接口中定义默认方法和静态方法。
- 默认方法:
default
修饰,可以带方法体,可以在实现类中重写,并可以通过接口的引用调用(接口.super) - 静态方法:
static
修饰,可以带方法体,不可以在实现类中重写,可以同接口名调用
public interface IPhoto {
//拍照方法
public void photo();
default void connection(){
System.out.println("我是接口中的默认链接");
}
static void stop(){
System.out.println("我是接口中的静态方法");
}
}
子类调用
@Override
public void connection() {
IPhoto.super.connection(); //调用接口中的默认方法
IPhoto.stop(); //调用接口中的静态方法
}
接口重名问题
关于多接口中重名默认方法处理的解决方案
1.如果在接口A和接口B中出现同名的默认方法,那么实现类必须实现这个默认方法。
2.如果某个类既继承了A类,又实现了B接口,而A类中包含的某个方法与B接口的某个默认方法同名,此时该类如果未重写该方法则调用的是A类中的方法
关于多接口名常量处理的解决方案
如上两段代码,运行是否正常?
第一段:在16行访问x时编译出错,因为此时编译器无法分辨x指代的是哪个接口中的x,如果只实现了一个接口可以这么写,但是此时必须修改为One.x或者Two.x
第二段:当继承的父类中成员属性和实现接口中的常量重名时,此时编译器依然无法分辨x指代的是谁,所以可以在TestOne类中定义自己的成员属性x,那么访问就没有问题了
接口的继承
接口可以实现多继承,即一个子接口可以同时继承多个父接口。
对于实现类而言,不仅需要实现当前接口的方法,还要实现接口继承的所有父接口的方法。
一个类可以继承自一个父类,同时实现多个接口。
对于接口继承的父接口如果出现重名的默认方法,则子接口也必须定义自己的默认方法。
接口与抽象类的区别
从语法层面讲
- 一个类只能继承一个抽象类,而一个类却可以实现多个接口
- 抽象类可以包含实现方法和普通任意类型字段,静态代码块以及静态方法。而接口只能定义常量和抽象方法(jdk1.7)
从设计层面讲
- 抽象类是对一种事物的抽象,即对类抽象,而接口是对行为的抽象。即抽象类是对整个类整体进行抽象,包括属性、行为,但是接口却是对类局部(行为)进行抽象
- 抽象类作为很多子类的父类,它是一种模板式设计。而接口是一种行为规范,它是一种辐射式设计