目录
抽象类
抽象类的由来
在抽取了子类(狗和狼)的共性之后,父类犬科定义吼叫功能,但是不知道犬科具体怎么叫,所以使用了抽象类,没有具体实现吼叫的内容,再例如正方形,三角形,圆形都是子类,而他们的父类是图形,那么现在图形想要实现计算面积的方法,没法执行具体的方法体实现,只能等子类继承时,确定了是什么具体图形的时候才能确定面积计算方式,那么这时候父类中的计算图形面积的方法只能定义为一个抽象方法。
在不断抽取子类共性内容的过程中,发现已经无法定义功能的具体实现,所以定义抽象类,只定义方法声明,不定义方法实现,让具体的子类继承时定义方法实现。
抽象类和抽象方法的定义格式
抽象方法:就是普通方法加上abstract关键字,然后去掉大括号,直接分号结束。
抽象类:抽象方法所在的类,必须是抽象类才行。在class之前写上abstract即可。
如何使用抽象类和抽象方法
1. 不能直接创建new抽象类对象。
2. 必须用一个子类来继承抽象父类。
3. 子类必须覆盖重写抽象父类当中所有的抽象方法。如果没有重写所有的抽象方法,那么
子类还是一个抽象类
覆盖重写(实现):子类去掉抽象方法的abstract关键字,然后补上方法体大括号。
4. 创建子类对象进行使用。
抽象类的特点
1:抽象方法只能定义在抽象类中,抽象类和抽象方法必须由abstract关键字修饰(可以描述类和方法,不可以描述变量)。
2:抽象方法只定义方法声明,并不定义方法实现。
3:抽象类不可以被创建对象(实例化),因为调用抽象方法时没意义,里面没方法体。
4:只有通过子类继承抽象类并覆盖了抽象类中的所有抽象方法后,该子类才可以实例化。否则,该子类还是一个抽象类。
abstract不能与什么关键字共存
abstract不能与关键词final和private共存,因为抽象类一定要被继承,而final是最终类,private如果修饰方法的话,方法就不能具体实现功能了
接口与抽象类的关系
当一个抽象类中全都是抽象方法,那么此时这个抽象类就是一个接口,用关键词interface代替abstract,接口是抽象类的一种特殊表现形式
抽象类不定义抽象方法可否
抽象类当中可以不定义抽象方法,意义只是不让该类创造实例对象
接口
接口是一种引用数据类型,把它理解成标准的公共规范,如我们生活中的USB接口
接口的定义和使用步骤
1. 接口不能直接使用,必须有一个“实现类”来“实现”该接口。
格式:
public class 实现类名称 implements 接口名称 {
// ...
}
2. 接口的实现类必须覆盖重写(实现)接口中所有的抽象方法。
实现:去掉abstract关键字,加上方法体大括号。
3. 创建实现类的对象,进行使用。
注意事项:
如果实现类并没有覆盖重写接口中所有的抽象方法,那么这个实现类自己就是抽象类。
接口中的内容
如果是Java 7,那么接口中可以包含的内容有:
1. 常量
2. 抽象方法
如果是Java 8,还可以额外包含有:
3. 默认方法
4. 静态方法
如果是Java 9,还可以额外包含有:
5. 私有方法
常量
接口当中也可以定义“成员变量”,但是必须使用public static final三个关键字进行修饰。
从效果上看,这其实就是接口的【常量】。
格式:
public static final 数据类型 常量名称 = 数据值;
备注:
一旦使用final关键字进行修饰,说明不可改变。
注意事项:
1. 接口当中的常量,可以省略public static final,注意:不写编译器默认添加,最好写上。
2. 接口当中的常量,必须进行赋值;不能不赋值。
3. 接口中常量的名称,使用完全大写的字母,用下划线进行分隔。(推荐命名规则)
4. 访问时直接用接口名称.常量的方式访问
抽象方法
在任何版本的Java中,接口都能定义抽象方法。
格式:
public abstract 返回值类型 方法名称(参数列表);
注意事项:
1. 接口当中的抽象方法,修饰符必须是两个固定的关键字:public abstract
2. 这两个关键字修饰符,可以选择性地省略。编译器会默认加上这两个关键字
3.方法的三要素:返回值类型,方法名,方法参数是随意些,但不能有方法体
举例:
// 这是一个抽象方法
public abstract void methodAbs1();
// 这也是抽象方法
abstract void methodAbs2();
// 这也是抽象方法
public void methodAbs3();
// 这也是抽象方法
void methodAbs4();
默认方法
格式:
public default 返回值类型 方法名称(参数列表) {
方法体
}
备注:接口当中的默认方法,可以解决接口升级的问题。
所谓升级,也就是说原先已经定义好一个接口,但是后面想加上一个方法,且此方法的内容是一致的,如果没有默认方法的功能,在接口中直接定义一个抽象方法,那么此接口的所有实现类都要去实现我们所定义的抽象方法,且方法体一致,那么有了这个默认方法之后,就可以直接将方法定义在接口中,子类进行可以直接调用。
1. 接口的默认方法,可以通过接口实现类对象,直接调用。
2. 接口的默认方法,也可以被接口实现类进行覆盖重写。
代码举例:
接口:
public interface MyInterfaceDefault { // 抽象方法 public abstract void methodAbs(); // 新添加的方法,改成默认方法 public default void methodDefault() { System.out.println("这是新添加的默认方法"); } }
实现类A
public class MyInterfaceDefaultA implements MyInterfaceDefault { @Override public void methodAbs() { System.out.println("实现了抽象方法,AAA"); } }
实现类B
public class MyInterfaceDefaultB implements MyInterfaceDefault { @Override public void methodAbs() { System.out.println("实现了抽象方法,BBB"); } @Override public void methodDefault() { System.out.println("实现类B覆盖重写了接口的默认方法"); } }
调用测试
public class Demo02Interface { public static void main(String[] args) { // 创建了实现类对象 MyInterfaceDefaultA a = new MyInterfaceDefaultA(); a.methodAbs(); // 调用抽象方法,实际运行的是右侧实现类。打印输出“实现了抽象方法,AAA” // 调用默认方法,如果实现类当中没有,会向上找接口 a.methodDefault(); // 这是新添加的默认方法 System.out.println("=========="); MyInterfaceDefaultB b = new MyInterfaceDefaultB(); b.methodAbs(); //打印输出“实现了抽象方法,BBB” b.methodDefault(); // 实现类B覆盖重写了接口的默认方法 } }
静态方法
格式:
public static 返回值类型 方法名称(参数列表) {
方法体
}
调用时,就跟类里定义的静态方法调用方式是一样的
注意事项:不能通过接口实现类的对象来调用接口当中的静态方法。
正确用法:通过接口名称,直接调用其中的静态方法。
格式:
接口名称.静态方法名(参数);
私有方法
解决问题场景:
代码如下:
public interface MyInterfacePrivateA { public default void methodDefault1() { System.out.println("默认方法1"); methodCommon(); } public default void methodDefault2() { System.out.println("默认方法2"); methodCommon(); } public default void methodCommon() { System.out.println("AAA"); System.out.println("BBB"); System.out.println("CCC"); } }
像上面的场景就会产生一个问题:
我们需要抽取一个共有方法,用来解决两个默认方法之间重复代码的问题。
但是这个共有方法只是为了两个默认方法服务,不应该让实现类使用,应该是私有化的。
解决方案:
从Java 9开始,接口当中允许定义私有方法。
1. 普通私有方法,解决多个默认方法之间重复代码问题
格式:
private 返回值类型 方法名称(参数列表) {
方法体
}
应对以上的问题,只要把methodCommon方法私有化即可,如下所示
private void methodCommon() { System.out.println("AAA"); System.out.println("BBB"); System.out.println("CCC"); }
这样就不会被接口的实现类调用到methodCommon这个方法,这个方法专门为了重复代码服务
2. 静态私有方法,解决多个静态方法之间重复代码问题
格式:
private static 返回值类型 方法名称(参数列表) {
方法体
}
静态私有方法与上面的普通私有方法应用类似,不过它抽取的是接口中静态方法中的共有内容
接口的特点
1.接口不能实例化对象(因为里面有抽象方法,你懂得)
2.子类必须覆盖所有的抽象方法后才能实例化,要不然子类中存在抽象方法,是一个抽象类
类与接口的关系
类与类之间存在着继承关系,类与接口中间存在的是实现关系。
继承用extends 实现用implements
接口多实现
接口的出现避免了单继承的局限性,类不可以多继承,但是可以多实现,接口最重要的体现是解决了多继承的弊端,将多继承这种机制通过多实现完成了,当初我上面说多继承时说的是当子类多继承父类时,父类可能有相同功能,子类调用会产生不确定性,不知道调用哪一个,而不知道调用哪一个的核心原因是因为两个父类中相同功能都有各自的主体,不确定运行哪个主体内容,而多实现接口的话,接口中的功能没有方法体,让子类自己去定义
使用接口的注意点
1. 接口是没有静态代码块或者构造方法的。
2. 一个类的直接父类是唯一的,但是一个类可以同时实现多个接口。
格式:
public class MyInterfaceImpl implements MyInterfaceA, MyInterfaceB {
// 覆盖重写所有抽象方法
}
3. 如果实现类所实现的多个接口当中,存在重复的抽象方法,那么只需要覆盖重写一次即可。
4. 如果实现类没有覆盖重写所有接口当中的所有抽象方法,那么实现类就必须是一个抽象类。
5. 如果实现类所实现的多个接口当中,存在重复的默认方法,那么实现类一定要对冲突的默认方法进行覆盖重写。
6. 一个类如果直接父类当中的方法和接口当中的默认方法产生了冲突,子类调用优先用父类当中的方法。(冲突指的是父类定义了public void method(){方法体}和接口中定义了public default void method(){方法体})
类与接口
1. 类与类之间是单继承的。直接父类只有一个。
2. 类与接口之间是多实现的。一个类可以实现多个接口。
3. 接口与接口之间是多继承的。一个接口可以继承多个接口
注意事项:
1. 多个父接口当中的抽象方法如果重复,没关系。
2. 多个父接口当中的默认方法如果重复,那么子接口必须进行默认方法的覆盖重写,【而且带着default关键字】。
抽象类和接口的区别
1:抽象类只能被继承,而且只能单继承。
接口需要被实现,而且可以多实现。
2:抽象类中可以定义非抽象方法,子类可以直接继承使用。
接口中都有抽象方法,需要子类去实现。
3:抽象类使用的是 is a 关系。
接口使用的 like a 关系。
4:抽象类的成员修饰符可以自定义。
接口中的成员修饰符是固定的。全都是public的。