一、抽象的概念
抽象是从众多的事物中抽取出共同的、本质性的特征,而舍弃其非本质的特征的过程。具体地说,抽象就是人们在实践的基础上,对于丰富的感性材料通过去粗取精、去伪存真、由此及彼、由表及里的加工制作,形成概念、判断、推理等思维形式,以反映事物的本质和规律的方法。(来自百度百科)
换句话说,抽象即抽取,抽取事物的共性。如:子类有正方形、三角形、圆形,其父类可抽取为图形。我们可以分别计算正方形、三角形、圆形的面积,却无法说出图形的面积计算方法,而在Java中,这个方法称为抽象方法。
二、抽象方法、抽象类
抽象方法:就是加上abstract关键字,然后去掉大括号,直接分号结束。
抽象类:抽象方法所在的类,必须是抽象类才行。在class之前写上abstract即可。
格式:
public abstract class Animal {
public abstract void eat();
}
那么,如何使用抽象类和抽象方法呢?有以下几点:
1、不能直接创建new抽象对象。
2、必须用一个子类来继承抽象父类。
3、子类必须覆盖重写抽象父类当中所有的抽象方法。
覆盖重写(实现):子类去掉抽象方法的abstract关键字,然后补上方法体的大括号。
4、创建子类对象进行使用。
代码示例:
public class Cat extends Animal {
@Override
public void eat() {
System.out.println("猫吃鱼");
}
}
public static void main(String[] args) {
Cat cat = new Cat();
cat.eat();
}
注意事项:
1、抽象类不能创建对象。
2、抽象类中,可以有构造方法,是供子类创建对象时,初始化父类成员使用的。
3、抽象类中,不一定包含抽象方法,但是有抽象方法的类必定是抽象类。
4、抽象类的子类,必须重写抽象父类中所有的抽象方法。否则,子类也需是抽象类。
三、接口的概念
在现实生活中,接口就是一种公共的规范标准。只要符合规范标准,大家就可以通用。如:USB接口。
而在Java中,接口就是多个类的公共规范。并且接口是一种引用数据类型,其中最重要的内容就是:抽象方法。
格式:
public interface 接口名称 {
//接口内容
}
备注:换成了关键字interface之后,编译生成的字节码文件仍然是:.java --> .class。
四、接口中可以包含的内容
如果是Java 7,那么接口中可以包含的内容有:
1、常量
2、抽象方法
如果是Java 8,还可以额外包含有:
3、默认方法
4、静态方法
如果是Java 9,还可以额外包含有:
5、私有方法
五、接口的抽象方法定义和使用
在任何版本的Java中,接口都能定义抽象方法。
格式:
public abstract 返回值类型 方法名称(参数列表);
注意:
1、接口当中的抽象方法,修饰符必须是两个固定的关键字:public abstract
2、这两个关键字修饰符,可以选择性地省略。
3、方法的三要素,可以随意定义。
代码示例:
public interface MyInterfaceAbstract {
//以下都是抽象方法
public abstract void methodAbs1();
abstract void methodAbs2();
public void methodAbs3();
void methodAbs4();
}
接口使用步骤:
1、接口不能直接使用,必须有一个“实现类”来“实现”该接口。
格式:
public class 实现类名称 implements 接口名称 {
//...
}
2、接口的实现类必须覆盖重写(实现)接口中所有的抽象方法。
实现:去掉abstract关键字,加上方法体大括号。
3、创建实现类的对象,进行使用。
注意:
如果实现类并没有覆盖重写接口中所有的抽象方法,那么这个实现类自己就必须是抽象类。
六、接口的默认方法定义和使用
从Java 8开始,接口里允许定义默认方法。
格式:
[public] default 返回值类型 方法名称(参数列表) {
方法体
}
备注:接口当中的默认方法,可以解决接口升级的问题。
1、接口的默认方法,可以通过接口实现类对象,直接调用。
2、接口的默认方法,也可以被接口实现类进行覆盖重写。
代码示例:
public interface MyinterfaceDefault {
//抽象方法
public abstract void methodAbs();
//新添加了一个抽象方法
//public abstract void methodAbs2();
public default void methodDefault(){
System.out.println("这是新添加的默认方法");
}
}
public class MyinterfaceDefaultA implements MyinterfaceDefault {
@Override
public void methodAbs() {
System.out.println("实现了抽象方法:AAA");
}
}
public class MyinterfaceDefaultB implements MyinterfaceDefault {
@Override
public void methodAbs() {
System.out.println("实现了抽象方法:BBB");
}
@Override
public void methodDefault() {
System.out.println("实现类B覆盖重写了接口的默认方法");
}
}
public static void main(String[] args) {
MyinterfaceDefaultA a = new MyinterfaceDefaultA();
a.methodAbs();//调用抽象方法,实际运行的是右侧的实现类
//调用默认方法,如果实现类当中没有,会向上找接口
a.methodDefault();//这是新添加的默认方法
System.out.println("###################");
MyinterfaceDefaultB b = new MyinterfaceDefaultB();
b.methodAbs();
b.methodDefault();//实现类B覆盖重写了接口的默认方法
}
七、接口的静态方法定义和使用
从Java 8开始,接口当中允许定义静态方法。
格式:
[public] static 返回值类型 方法名称(参数列表){
方法体
}
提示:就是将abstract或者default换成static即可,带上方法体。
注意:不能通过接口实现类的对象来调用接口当中的静态方法。
正确用法:通过接口名称,直接调用其中的静态方法。
格式:接口名称.静态方法名(参数);
八、接口的私有方法定义和使用
问题描述:
当我们需要抽取一个共有方法,用来解决几个默认方法之间重复代码的问题。但是这个共有方法不应该让实现类使用,应该是私有化的。
解决方案:
从Java 9开始,接口当中允许定义私有方法。
1、普通私有方法,解决多个默认方法之间重复代码问题
格式:
private 返回值类型 方法名称(参数列表) {
方法体
}
2、静态私有方法,解决多个静态方法之间重复代码问题
格式:
private static 返回值类型 方法名称(参数列表) {
方法体
}
代码示例:
public interface MyinterfacePrivateA {
public default void methodDefault1(){
System.out.println("默认方法1");
methodCommon();
}
public default void methodDefault2(){
System.out.println("默认方法2");
methodCommon();
}
private void methodCommon(){
System.out.println("AAA");
System.out.println("BBB");
System.out.println("CCC");
}
}
public interface MyinterfacePrivateB {
public static void methodStatic1(){
System.out.println("静态方法1");
methodStaticCommon();
}
public static void methodStatic2(){
System.out.println("静态方法2");
methodStaticCommon();
}
private static void methodStaticCommon(){
System.out.println("AAA");
System.out.println("BBB");
System.out.println("CCC");
}
}
MyinterfacePrivateB.methodStatic1();
MyinterfacePrivateB.methodStatic2();
注意:private的方法只有接口自己才能调用,不能被实现类或别人使用。
九、接口的常量定义和使用
接口当中也可以定义“成员变量”,但是必须使用public static final三个关键字进行修饰。从效果上看,这其实就是接口的【常量】。
格式:
public static final 数据类型 常量名称 = 数据值;
备注:一旦使用final关键字进行修饰,说明不可改变。
注意:
1、接口当中的常量,可以省略public static final。
2、接口当中的常量,必须进行赋值。
3、接口中常量的名称,使用完全大写的字母,用下划线进行分隔。(推荐命名规则)
代码示例:
public interface MyinterfaceConst {
//这其实就是一个常量,一旦赋值,不可以修改
public static final int NUM_OF_MY_CLASS = 30;
}
public static void main(String[] args) {
//访问接口当中的常量
System.out.println(MyinterfaceConst.NUM_OF_MY_CLASS);
}
十、使用接口的注意事项
1、接口是没有静态代码块或者构造方法的。
2、一个类的直接父类是唯一的,但是一个类可以同时实现多个接口。
格式:
public class MyInterfaceImpl implements MyInterfaceA,MyInterfaceB {
//覆盖重写所有抽象方法
}
3、如果实现类所实现的多个接口当中,存在重复的抽象方法,那么只需要覆盖重写一次即可。
4、如果实现类没有覆盖重写所有接口当中的所有抽象方法,那么实现类就必须是一个抽象类。
5、如果实现类所实现的多个接口当中,存在重复的默认方法,那么实现类一定要对冲突的默认方法进行覆盖重写。
6、一个类如果直接父类当中的方法,和接口当中的默认方法产生了冲突,优先用父类当中的方法。
十一、接口之间的多继承
1、类与类之间是单继承的,直接父类只有一个。
2、类与接口之间是多实现的,一个类可以实现多个接口。
3、接口与接口之间是多继承的。
public interface MyInterface extends MyInterfaceA,MyInterfaceB{
//...
}
注意:
1、多个父接口当中的抽象方法可以重复。(子接口就看做一个方法)
2、多个父接口当中的默认方法如果重复,那么子接口必须进行默认方法的覆盖重写,【而且带着default关键字】。