1. 接口
1.1 概述
接口,是Java语言中一种引用类型,是方法的集合,如果说类的内部封装了成员变量、构造方法和成员方法,那么接口的内部主要就是封装了方法,包含抽象方法,默认方法和静态方法,私有方法。
接口的定义,它与定义类方式相似,但是使用interface关键字。它也会被编译成.class文件,但一定要明确它并不是类,而是另外一种引用数据类型。
接口的使用,它不能创建对象,但是可以被实现(implements,类似于被继承)。一个实现接口的类(可以看做是接口的子类),需要实现接口中所有的抽象方法,创建该类对象,就可以调用方法了,否则它必须是一个抽象类。
1.2 定义格式

含有抽象方法
- 抽象方法:使用abstract关键字修饰,可以省略,没有方法体。该方法供子类实现使用。
代码如下:public interface InterFaceName{ public abstract void method(); } 
含有默认方法和静态方法
- 默认方法:使用default修饰,不可省略,供子类调用或者子类重写。
 - 静态方法:使用static修饰,供接口直接调用。
 
代码如下:
public interface InterFaceName{
    public default void method(){
    // 执行语句
    }
    public static void method2(){
    // 执行语句
    }
}
含有私有方法和私有静态方法
- 私有方法:使用private修饰,供接口中的默认方法或者静态方法调用。
 
代码如下:
public interface InterFaceName{
    private void method(){
        // 执行语句
    }
}
1.3 基本的实现
类与接口的关系为实现关系,即类实现接口,该类可以称为接口的实现类,也可以称为接口的子类。实现的动作类似继承,格式相似,只是关键字不同,实现使用implements关键字。
非抽象子类实现接口:
- 必须重写接口中所有抽象方法。
 - 继承了接口的默认方法,即可以直接调用,也可以重写。
格式如下:
 
代码示例:
// 定义接口类
public interface LiveAble {
    // 定义抽象方法
    public abstract void eat();
    public abstract void sleep();
}
// 定义实现类
public class Animal implements LiveAble{
    @Override
    public void eat(){
        System.out.println("吃东西");
    }
    @Override
    public void sleep(){
        System.out.println("睡觉");
    }
}
// 定义测试类
public class InterfaceDemo {
    public static void main(String[] args) {
        // 创建子类对象
        Animal a = new Animal();
        a.eat();
        a.sleep();
    }
}
默认方法的使用
可以继承,可以重写,但只能通过实现类的对象来调用。接口当中的默认方法,可以解决接口升级的问题。
- 实现类可以继承接口类的默认方法,也可以自己重写默认方法。
 
代码示例如下:
// 定义接口类
public interface LiveAble {
    // 定义抽象方法
    public abstract void eat();
    public abstract void sleep();
    public default void fly(){ // 定义默认方法
        System.out.println("飞行");
    }
}
// 定义实现类
public class Animal implements LiveAble{
    @Override
    public void fly(){ // 重写接口中的默认方法,也可以不重写,继承接口类中的默认方法
        System.out.println("自由自在的飞!");
    }
}
// 定义测试类
public class InterfaceDemo {
    public static void main(String[] args) {
        // 创建子类对象
        Animal a = new Animal();
        a.eat();
        a.sleep();
        a.fly(); // 如果实现类没有重写默认方法,则执行的是继承自接口类的默认方法,重写则执行实现类重写后的方法
    }
}
静态方法的使用
静态与.class文件相关,只能使用接口名调用,不能通过实现类的类名或者实现类的对象调用
示例如下:
// 接口类
public interface LiveAble {
    public static void run(){
        System.out.println("这是静态方法,只能通过接口类调用");
    }
}
// 实现类无法继承或重写接口类静态方法
// 测试类
public class InterfaceDemo {
    public static void main(String[] args) {
        // 创建子类对象
        Animal a = new Animal();
    // a.run(); 报错,无法通过实现类调用静态方法
        LiveAble.run(); // 接口类的静态方法只能通过接口类来调用
    }
}
私有方法的使用
- 私有方法:只有默认方法可以调用
 - 私有静态方法:默认方法和静态方法可以调用。
如果一个接口中有多个默认方法且方法中有代码重复内容,那么可以抽取出来,封装到私有方法中,供默认方法去调用。私有方法是对默认方法和静态方法的辅助。public interface LiveAble { default void func(){ func1(); func2(); } private void func1(){ System.out.println("跑起来~~~"); } private void func2(){ System.out.println("跑起来~~~"); } }接口的常量定义和使用
接口当中也可可以定义”成员变量”,但是必须使用public static final三个关键字进行修饰。可以认为就是接口的[常量]。
格式:public static final 数据类型 常量名称 = 数据值; 一旦使用final关键字进行修饰,说明不可改变。
注意事项: 
- 接口中的常量,可以省略public static final。
 - 接口中的常量,必须进行赋值,不能不赋值。
 - 接口中常量的名称规则:使用完全大写的字母,单词间用下划线进行分隔。
 
接口内容小结
在Java 9+版本中,接口的内容可以有:
- 成员变量其实是常量。
- 格式:[public] [sttaic] [final] 数据类型 常量名称 = 数据值;
 - 注意:常量必须进行赋值,且一旦赋值不能改变。常量名称完全大写,用下划线分隔。
 
 - 接口中最重要的就是抽象方法。
- 格式:[public] [abstract] 返回值类型 方法名称(参数列表);
 - 注意:实现类必须覆盖重写接口的所有抽象方法,除非实现类是抽象类。
 
 - 从Java 8开始,接口里允许定义默认方法。
- 格式:[public] default 返回值类型 方法名称(参数列表){方法体}
 - 注意:默认方法也可以被覆盖重写
 
 - 从Java 8开始,接口里允许定义静态方法。
- 格式:[public] static 返回值类型 方法名称(参数列表){方法体}
 - 注意:应该通过接口名称进行调用,无法通过实现类对象调用接口静态方法。
 
 - 从Java 9开始,接口里允许定义私有方法。
- 普通私有方法:private 返回值类型 方法名称(参数列表){方法体}
 - 静态私有方法:private static 返回值类型 方法名称(参数列表){方法体}
 - 注意:private方法只要接口自己才能调用,无法被实现类或者别人使用。
 
 
接口的多实现
在继承体系中,一个类只能继承一个父类。而对于接口而言,一个类是可以实现多个接口的,这叫做接口的多实现。并且,一个类能继承一个父类,同时实现多个接口。
格式如下:
使用接口的时候,需注意:
- 接口是没有静态代码块或者构造方法的。
 - 一个类的直接父类是唯一的,但是一个类可以同时实现多个接口的。
格式:
public class MyIntfaceImpl implements MyInterfaceA,MyInterfaceB{
// 覆盖重写所有抽象方法
} - 如果实现类实现的多个接口中,存在重复的抽象方法,只需覆盖重写一次即可。
 - 如果实现类没有覆盖重写所以接口的所有抽象方法,那么实现类必须是抽象类。
 - 如果实现类所实现的多个接口当中,存在重复的默认方法,那么实现类一定要对冲突的默认方法进行覆盖重写。
 - 一个类如果直接父类当中的方法,和接口当中的默认方法产生了冲突,子类优先用父类当中的方法。
 
接口的多继承
一个接口能继承另一个或者多个接口。接口的继承使用extends关键字,子接口继承父接口的方法。如果父接口中的默认方法有重名,那么子接口需要重写一次。
// 父接口A
public interface MyInterfaceA {
    public abstract void methodA();
    public abstract void methodCommon();
    public default void methodDefault(){
        System.out.println("接口A默认方法");
    }
}
// 父接口B
public interface MyInterfaceB {
    public abstract void methodB();
    public abstract void methodCommon();
    public default void methodDefault(){
        System.out.println("接口B默认方法");
    }
}
// 子接口
public interface MyInterface extends MyInterfaceA,MyInterfaceB{
    public abstract void method();
    @Override
    default void methodDefault() { // 重写父接口的默认方法
    }
}
- 类与类之间是单继承的。直接父类只有一个
 - 类与接口之间是多实现的。一个类可以实现多个接口。
 - 接口与接口之间是多继承的。
 
注意事项:
- 多个父接口当中的抽象方法如果重复,没关系
 - 多个父接口中的默认方法如果重复,那么子接口必须进行默认方法的覆盖重写,[而且带着default关键字
 
2.多态
- 多态:是指同一行为,具有多个不同表现形式。
 
2.1多态的体现
多态的体现格式:父类名称 对象名 = new 子类名称();
代码中体现多态性,其实就是一句话:父类引用指向子类对象
成员变量在多态中的规则
// 父类
public class Fu {
    int num = 10;
    public void showNum(){
        System.out.println(num);
    }
}
// 子类
public class Zi extends Fu {
    int num = 20;
    @Override
    public void showNum(){
        System.out.println(num);
    }
}
// 测试类
public class DemoMulti01 {
    public static void main(String[] args) {
        Fu obj = new Zi();
        System.out.println(obj.num); // 直接访问成员变量时,左边是谁就先访问谁的
        // 子类没有覆盖重写,就是父:10
        // 子类如果覆盖重写,就是子:20
        obj.showNum();
    }
}
注意事项:
- 直接通过对象名称访问成员变量:看等号左边是谁优先访问谁,没有则向上查找。
 - 间接通过成员方法来访问成员变量:看该方法属于谁优先访问谁,没有则向上找。
 
多态中成员方法的使用特点
当使用多态方式调用方法时,首先检查父类中是否有该方法,如果没有,则编译错误;如果有,执行的是子类重写后方法。
// 父类
public class Fu {
    int num = 10;
    public void showNum(){
        System.out.println(num);
    }
    public void method(){
        System.out.println("父类方法");
    }
    public void methodFu(){
        System.out.println("父类特有方法");
    }
}
// 子类
public class Zi extends Fu {
    int num = 20;
    @Override
    public void showNum(){
        System.out.println(num);
    }
    @Override
    public void method(){
        System.out.println("子类方法");
    }
    public void methodZi(){
        System.out.println("子类特有方法");
    }
}
// 测试类
/*
在多态的代码中,成员方法的访问规则是:
    看new的是谁,就优先用谁,没有则向上找。
成员方法口诀:编译看左边,运行看右边。
对比一下:
成员变量:编译看左边,运行还看左边。
成员方法:编译看左边,运行看右边。
 */
public class Demo02MultiMethod {
    public static void main(String[] args) {
            Fu obj = new Zi(); // 多态
            // 运行看右边:左边是Fu,Fu有method方法,所以编译通过,但是运行时看右边是Zi,故运行子类方法。
            obj.method(); // 父子都有,优先用子
            // 运行看右:左边是Fu,Fu有methodFu方法,编译通过,但运行是看右Zi,子类没有该方法,就向上找到父类,然后执行。
            obj.methodFu(); // 子类没有,向上找到父类有
            // 编译看左边:左边是Fu,Fu中没有methodZi方法,所以编译报错。
            // obj.methodZi(); 错误写法!
    }
}
2.2 多态的好处
可以使程序编写的更简单,并且有良好的扩展性。
2.3 引用类型转换
多态的转型分为向上转型和向下转型两种:
向上转型
- 概述:多态本身就子类类型向父类类型向上转换的过程,这个过程是默认的。
 - 使用格式:父类名称 变量名 = new 子类类型(); 如 Animal a = new Cat();
 - 含义:右侧创建一个子类对象,把它当做父类来看待使用。如上创建了一只猫,当做动物看待,没问题。
 - 注意事项:向上转型一定是安全的,从小范围转向了大范围,从小范围的猫,向上转换成为更大范围的动物。
 - 类似于:double num = 100; // 正确, int–> double, 自动类型转换。
 
向下转型
- 概述:父类类型向子类类型向下转换的过程,这个过程是强制的。一个已经向上转型的子类对象,将父类引用转为子类引用,可以使用强制类型转换的格式,便是向下转型。
 - 格式:子类名称 对象名 = (子类名称)父类对象;如 Cat cat = (Cat)a;
 - 含义:将父类对象,[还原]成为本来的子类对象。
 
注意事项:
- 必须保证对象本来创建的时候,就是猫,才能向下转型成为猫。
 - 如果对象创建时本来就不是猫,现在非要向下转型为猫,会报错ClassCastException
 - 类似于: int num = (int)10.0; // 可以 int num = (int)10.5 //不可以,精度损失。
 
instanceof关键字
为了避免ClassCastException的发生,Java提供了 instanceof 关键字,给引用变量做类型的校验。
格式:变量名 instanceof 数据类型; 如果变量属于该数据类型,返回true,反之返回false。
// 父类
public abstract class Animal {
    public abstract void eat();
}
// 子类1
public class Cat extends Animal{
    @Override
    public void eat() {
        System.out.println("猫吃鱼");
    }
    public void jump(){
        System.out.println("猫跳跳");
    }
}
// 子类2
public class Dog extends Animal{
    @Override
    public void eat() {
        System.out.println("狗吃shit");
    }
    public void watchHouse(){
        System.out.println("看家");
    }
}
// 测试类
public class MainMethod {
    public static void main(String[] args) {
        Animal animal = new Dog();
        animal.eat();
        getAnimal(new Cat());
    }
    public static void getAnimal(Animal animal){
        if(animal instanceof Dog){
            Dog dog = (Dog)animal;
            dog.watchHouse();
        }
        if(animal instanceof Cat){
            Cat cat = (Cat)animal;
            cat.jump();
        }
    }
}
3. 接口多态的综合案例
定义USB接口,具备最基本的开启功能和关闭功能。鼠标和键盘要想能在电脑上使用,那么鼠标和键盘也必须遵守USB规范,实现USB接口,否则鼠标和键盘的生产出来也无法使用。
3.1 案例分析
进行描述笔记本类,实现笔记本使用USB鼠标、USB键盘
- USB接口,包含开启功能、关闭功能
 - 笔记本类,包含运行功能、关机功能、使用USB设备功能
 - 鼠标类,要实现USB接口,并具备点击的方法
 - 键盘类,要实现USB接口,具备敲击的方法
 
具体代码实现如下:
// 接口类
public interface USB {
    public abstract void open(); // 打开设备
    public abstract void close(); // 关闭设备
}
// 笔记本类
public class Laptop {
    public void powerOn(){
        System.out.println("笔记本电脑开机");
    }
    public void powerOff(){
        System.out.println("笔记本电脑关机");
    }
    // 使用USB设备的方法,使用接口作为方法参数
    public void usbDevice(USB usb){
        usb.open();
        if(usb instanceof Mouse){
            Mouse mouse = (Mouse)usb; // 向下转型
            mouse.click();
        }else if (usb instanceof Keyboard){
            Keyboard keyboard = (Keyboard)usb;
            keyboard.click();
        }
        usb.close();
    }
}
// 鼠标类
// 鼠标就是一个USB设备
public class Mouse implements USB{
    @Override
    public void open() {
        System.out.println("打开鼠标");
    }
    @Override
    public void close() {
        System.out.println("关闭鼠标");
    }
    public void click(){
        System.out.println("鼠标点击");
    }
}
// 键盘类
// 键盘也是一个USB设备
public class Keyboard implements USB{
    @Override
    public void open() {
        System.out.println("打开键盘");
    }
    @Override
    public void close() {
        System.out.println("关闭键盘");
    }
    public void click(){
        System.out.println("键盘输入");
    }
}
// 测试类
public class DemoMain {
    public static void main(String[] args) {
        // 首先创建一个电脑
        Laptop laptop = new Laptop();
        laptop.powerOn();
        // 准备一个鼠标,供电脑使用
        USB usbMouse = new Mouse(); // 首先向上转型
        laptop.usbDevice(usbMouse);
        // 创建一个USB键盘
        Keyboard keyboard = new Keyboard(); //没有使用多态写法
        // 方法参数是USB类型,传递进去的是实现类对象
        laptop.usbDevice(keyboard); // 正确写法! 自动发生了向上转型!
        laptop.powerOff();
    }
}
该案例主要练习对接口的基本使用,对象的上下转型以及接口作为对象参数的使用。
今日总结
- 接口的概述以及定义格式,以及实现接口的格式。interface关键字,implements关键字
 - 接口中的抽象方法,默认方法和静态方法,私有方法和私有静态方法的各自使用特点。
 - 接口中成员方法,成员变量的特点。
 - 接口的多继承。
 - 多态的概念以及前提。
 - 多态中的向上向下转型格式以及实际使用方法。
 - instanceof关键字。