一. 接口概念
1.1 接口是一类事物
从面向对象程序设计思想上看,Java中接口interface和类class 都是用来描述一类事物的工具。类关注事物的属性和行为两个方面,但是接口通常只关注事物的行为单方面。
1.2 接口是能力集合
从编程业务角度看,接口是一系列功能的集合,因为接口中通常只会声明方法,而这些方法,都是一个一个的功能,所以可以理解接口是能力的集合。
1.3 接口是一种规范
从代码设计角度看,接口是一种规范和准则,是一套严格的标准,系统各个模块通过接口连接,并实现解耦 ,软件设计中,接口通常作为设计体系结构的工具。
接口定义:
//定义接口public interface Usb {//静态常量public static final String VERSION= "3.0";//抽象方法public abstract void transData();//静态方法public static void stop(){System.out.println("stop....");}//默认方法public default void flash(){System.out.println("flash。。。");}}
2. 接口与实现类
类和类上下级是继承关系,类和接口上下级是 实现关系 ,类去实现接口。接口是比抽象类还抽象的事物,所以接口也不能实例化本身的实例, 单独定义一个接口没有意义,接口必须用自己的实现类,才可以使用。
2.1 实现语法
class 实现类名 extends 父类名 implements 接口名1,接口名2,接口名N{// 1. 这类本身该有的东西// 2. 必须重写全部接口的抽象方法。}实现类// 定义一个类public class UPan implements Usb {@Overridepublic void transData() {System.out.println("U盘"+ Usb.VERSION + "协议传数据...");}}
2.2 类和类关系
类是单继承的,一个类最多只能直接继承一个父类 ,但是可以多级继承。
2.3 类和接口的关系
类是多实现的,一个类可以实现多个接口,但是要重写全部的抽象方法。
三. 接口与抽象类的区别
3.1 相同点
- 都可以定义抽象方法
- 都不可以实例化本身实例
-
3.2 不同点
声明关键字不同接口 intercace class
- 定义内容不同( 接口不可以定义 实例属性 构造器 代码块 )
- 抽象类中抽象方法 需要显示声明。
体会: 有时候接口抽象类甚至是普通类都可以完成设计,但是我们需要选择一个最好的方案
4.接口多态
4.1 什么是接口多态
和类多态一样,声明接口 实例化 实现类对象。由于类是多实现,接口引用类型较多,通过不同的接口引用调用方法,只能调用到该接口中声明的方法。
public class TestUsb {public static void main(String[] args) {//多态Usb usb = new UPan();usb.transData();usb.save();//报错调用不了,Usb接口中没有这个方法System.out.println("------------------------");//多态Storage usb2 = new UPan();usb2.save();usb2.transData();//报错调用不了,Storage接口中没有这个方法}}
4.2 接口向下转型
由于多态问题,导致接口引用无法调用实现类特有方法,可以向下转型,以便调用实现类特有方法
public class TestUsb {public static void main(String[] args) {//多态1Usb usb = new UPan();usb.transData();// usb.save();//报错调用不了,Usb接口中没有这个方法 , 通过向下转型来解决UPan xx = (UPan) usb;xx.save(); //这里使用 xx 就可以调用特有方法。}}
5. 什么是接口继承
和类一样,子接口通过继承获得父接口中的抽象方法,但是接口的继承是是多继承,一个接口可以同时继承多个父接口。
interface IA{void a();}interface IB{void b();}interface IC extends IB,IA {void c();}class X implements IC { // 你看这里重写了 3个抽象方法@Overridepublic void a() {}@Overridepublic void b() {}@Overridepublic void c() {}}
6.接口回调
指的是,把接口设计为方法的参数,方法的参数是一个接口类。它出现的场景是,当我们想封装一个通用的方法,但是发现这个方法不能完全封装,有些步骤可以封装,但是有些步骤又存在差异不可封装,这是咱们可以把可以封装的逻辑依次编写,不可封装的部分,通过参数来执行方法解决
//回调案例public class CallBackDemo {public static void main(String[] args) {Person person = new Person();person.speak( new LSLife() );}}//抽取的一个接口,讲述一个人的经历,具体不详细interface Life{public void tel();}class ZSLife implements Life{@Overridepublic void tel() {System.out.println("生于1890");System.out.println("年少有为");System.out.println("毕业蓝翔技校");System.out.println("于2080....");}}class LSLife implements Life{@Overridepublic void tel() {System.out.println("生于1891");System.out.println("年少有为个屁");System.out.println("毕业蓝翔技校");System.out.println("于2081....");}}class Person{//我们想封装的一个通用功能 ,有些流程是固定,但是有部分是不确定的。public void speak( Life obj ){System.out.println("各位亲朋好友大家好");//不可确定的obj.tel();System.out.println("谢谢各位,开席!");}}
