当前位置: 首页 > news >正文

Java SE(10)——抽象类接口

1.抽象类

1.1 概念

在之前讲Java SE(6)——类和对象(一)的时候说过,所有的对象都可以通过类来抽象。但是反过来,并不是说所有的类都是用来抽象一个具体的对象。如果一个类本身没有足够的信息来描述一个具体的对象,而是用于定义一个模板,为子类提供通用的属性和方法,这样的类就是抽象类
在这里插入图片描述

1.2 语法规则

在Java中,被abstract修饰的类就是抽象类;被abstract修饰的方法就是抽象方法

//抽象类
public abstract class AbstractClass {//抽象方法public abstract void abstractMethod();
}
  • 1.抽象类不能直接实例化,它只能作为其他类的父类 在这里插入图片描述
  • 2.抽象方法是一种没有具体实现的方法,即没有方法体 在这里插入图片描述
  • 3.抽象类中也可以定义普通成员变量/普通成员方法/构造方法。换言之,抽象类中不一定有抽象方法,但是抽象方法所在的类一定是抽象类
    在这里插入图片描述
    在这里插入图片描述
  • 4.abstract只能修饰类和方法,没有抽象变量这个概念 在这里插入图片描述
  • 5.抽象方法不能用private/static/final修饰 在这里插入图片描述
问题:既然抽象类无法直接实例化对象,那么抽象类真正的作用是什么?
在上面讲概念和语法规则的时候说过,抽象类只能作为父类被其他类继承,给子类提供一个模板并由子类来实现。

1.3 抽象类的实现

public abstract class Shape {//普通成员变量public int width;public int length;//普通方法public void hello(){System.out.println("hello shape");}//构造方法public Shape(int width, int length) {this.width = width;this.length = length;}public abstract void draw();
}
public class Circle extends Shape {public Circle(int width, int length) {super(width, length);}@Overridepublic void draw() {System.out.println("draw circle:" + this.width + " * " + this.length);}@Overridepublic void hello(){System.out.println("hello circle");}
}
public class Test {public static void main(String[] args) {//无法直接实例化抽象类/*Shape shape = new Shape();*/Shape circle = new Circle(10,10);circle.draw();circle.hello();}
}

运行结果:
draw circle:10 * 10
hello circle

  • 1.当某一个类(称为实现类)继承抽象类时,该实现类必须重写抽象类中的所有抽象方法 (这也是抽象方法不能使用private/static/final修饰的根本原因)在这里插入图片描述
  • 2.如果是抽象类A继承抽象类B,则不需要重写父类中的抽象方法。直到有实现类C来继承抽象类A,那么实现类C要重写A和B中的所有抽象方法 在这里插入图片描述

1.4 意义

仔细观察上述代码,抽象类和实现类之间构建了继承关系,发生了向上转型/方法重写。但是普通类和普通类之间好像也可以完成上述操作,换言之,抽象类能完成的功能普通类也可以。那么抽象类存在的意义是什么?
还是以上述代码为例,此时的需求是画一个circle。假设父类使用普通类,如果用户一不小心将父类直接实例化,那么此时调用draw()方法就无法画一个circle出来;再假设父类使用抽象类,如果用户直接实例化父类是会报错的,而且子类如果不重写抽象方法也是会报错的,这就是抽象类在提醒用户。使用抽象类相较于使用普通类,在特定场景下多了一层校验效果

2.接口

2.1 概念

在日常生活中,经常听到接口这个东西。比如:电脑的USB-A接口,既可以连接键盘,又可以连接鼠标、耳机等等,只要是适配UEB协议的设备都可以连接
在这里插入图片描述
通过这些例子可以简述一下接口的作用:接口就是公共的行为规范标准。在实现某些功能时只要符合该规范标准就行了
在Java中,接口可以看作是多个类的公共规范,是一种引用数据类型

2.2 语法规则

定义接口需要借助interface关键字

public interface 接口名称{
}
  • 1.接口中的方法默认使用public abstract修饰,所以定义方法时,建议不要再手动添加修饰词
    在这里插入图片描述
  • 2.接口中的变量默认使用public static final修饰 在这里插入图片描述
  • 3.接口中不能定义构造方法 在这里插入图片描述
  • 4.在JDK8及以后,接口中可以定义静态方法和default方法 在这里插入图片描述
    default方法时接口中独有的方法,等会儿仔细说说default方法的作用

2.3 接口的实现

接口和抽象类一样,也不能直接实例化对象,只能由某个具体的类来实现。并且在Java SE(8)——继承中讲过,一个类只能拥有一个父类,但是一个类可以实现多个接口

类实现接口使用的是implements关键字

2.3.1 实现单接口

需求:实现电脑使用Usb设备(鼠标、键盘)

  • IUsb接口:包含打开设备、关闭设备的功能
  • Mouse类:实现IUsb接口,并具备点击功能
  • KeyBoard类:实现IUsb接口,并具备输入功能
  • Computer类:包含开机功能、关机功能和使用IUsb设备功能
//IUsb接口
public interface IUsb {void openDevice();void closeDevice();
}
//Mouse类实现IUsb接口
public class Mouse implements IUsb {@Overridepublic void openDevice() {System.out.println("Mouse open");}@Overridepublic void closeDevice() {System.out.println("Mouse close");}public void click(){System.out.println("Mouse click");}
}
//KeyBoard类实现IUsb接口
public class KeyBoard implements IUsb {@Overridepublic void openDevice() {System.out.println("KeyBoard open");}@Overridepublic void closeDevice() {System.out.println("KeyBoard close");}public void input(){System.out.println("KeyBoard input");}
}
//电脑类,使用Use设备(鼠标、键盘)
public class Computer {public void open(){System.out.println("Computer open");}public void useUsb(IUsb iusb){iusb.openDevice();if (iusb instanceof KeyBoard){//向下转型KeyBoard keyBoard = (KeyBoard)iusb;keyBoard.input();}if (iusb instanceof Mouse){//向下转型Mouse mouse = (Mouse)iusb;mouse.click();}iusb.closeDevice();}public void close(){System.out.println("Computer close");}
}
//测试类
public class Test {public static void main(String[] args) {Computer computer = new Computer();computer.open();computer.useUsb(new Mouse());computer.useUsb(new KeyBoard());computer.close();}
}

运行结果:
Computer open
Mouse open
Mouse click
Mouse close
KeyBoard open
KeyBoard input
KeyBoard close
Computer close

2.3.2 实现多接口

public interface IRunning {void run();
}
public interface ISwimming {void swim();
}
public class Animal {public String name;public Animal(String name){this.name = name;}
}
//青蛙既能跑又能游
public class Frog extends Animal implements IRunning,ISwimming{public Frog(String name) {super(name);}@Overridepublic void run() {System.out.println(this.name + " is running");}@Overridepublic void swim() {System.out.println(this.name + " is swimming");}public void act(){this.run();this.swim();}
}
public class Test {public static void main(String[] args) {Frog frog = new Frog("Frog");frog.act();}
}

运行结果:
Frog is running
Frog is swimming

2.4 接口的继承

在Java中,类和类之间是单继承的,⼀个类可以实现多个接口,接口与接口之间可以多继承

即:用接口可以达到多继承的目的

public interface IRunning {void run();
}
public interface ISwimming {void swim();
}
//Act接口继承IRunning和ISwimming接口
public interface Act extends IRunning,ISwimming{@Overridevoid run();@Overridevoid swim();
}
public class Animal {public String name;public Animal(String name){this.name = name;}
}
//青蛙既能跑又能游
public class Frog extends Animal implements Act{public Frog(String name) {super(name);}@Overridepublic void run() {System.out.println(this.name + " is running");}@Overridepublic void swim() {System.out.println(this.name + " is swimming");}public void act(){this.run();this.swim();}
}
public class Test {public static void main(String[] args) {Frog frog = new Frog("Frog");frog.act();}
}

运行结果不变

2.5 default方法的作用

  • 默认实现:允许拥有具体的实现。实现类可以选择是否重写该方法
  • 多重继承的解决方法:当某个具体类实现了非常多的接口时,接口之间可能会存在同名的方法。通过default方法可以解决同名方法的冲突 ```java public interface IA {
    default void func(){
    System.out.println(“IA func”);
    } } public interface IB {
    default void func(){
    System.out.println(“IB func”);
    } } public class Demo implements IA,IB {
    @Override
    public void func() {
    //调用IA中的默认方法
    IB.super.func();
    } } public class Test {
    public static void main(String[] args) {
    Demo demo = new Demo();
    demo.func();
    } } ````在上述代码中必须在Demo类中重写func()方法才能解决同名冲突问题,所谓的重写就是选择调用IA还是IB中的default方法,并没有改变方法的原有实现。`如果IA和IB中是public
    abstract修饰的方法,没有具体实现,只能由Demo类来实现,那么必然有一个接口中的func()方法无法被重写
  • 向后兼容性:default方法使得在现有的接口中添加新方法成为可能,而不会破坏已有的实现类。如果没有default方法,添加新方法到接口会导致所有实现类都需要实现这个新方法,这会破坏向后兼容性

2.5 接口和抽象类的区别

核心区别:抽象类中可以包含普通方法和普通字段(变量),这样的普通方法和字段可以被子类直接使用(不必重写);而接口中不能包含普通方法,子类必须重写所有的抽象方法

如之前写的 Animal 例子。此处的Animal中包含⼀个name这样的属性,这个属性在任何子类中都是存在的。因此下面的Animal只能作为⼀个抽象类,而不应该成为一个接口

public class Animal {public String name;public Animal(String name){this.name = name;}
}

相关文章:

  • Python实例题:Python快速获取斗图表情
  • 【python】Calculate the Angle of a Triangle
  • 为啥大模型一般将kv进行缓存,而q不需要
  • 量化解析美英协议的非对称冲击:多因子模型与波动率曲面重构
  • Spring Boot初级教程:从零搭建企业级Java应用
  • Linux 学习笔记2
  • Spark,在shell中运行RDD程序
  • 数据结构(1)复杂度
  • Git回顾
  • 关于VScode的调试
  • DVWA靶场保姆级通关教程--07SQL注入(上)
  • JVM之内存管理(一)
  • Servlet、HttpServlet 和 DispatcherServlet 区别与关系
  • 鸿蒙开发:dialog库做了一些优化
  • htmlUnit和Selenium的区别以及使用BrowserMobProxy捕获网络请求
  • 住宅IP的深度解析与合理运用
  • 聊聊Spring AI autoconfigure模块的拆分
  • 在线工具源码_字典查询_汉语词典_成语查询_择吉黄历等255个工具数百万数据 养站神器,安装教程
  • DeepSeek“智”造:解锁旅游行业新玩法
  • stm32F103芯片 实现PID算法控制温度例程
  • 总粉丝破亿!当网络大V遇见硬核科技,互联网时代如何书写上海故事?
  • 洛杉矶奥组委确认2028年奥运会和残奥会开闭幕式场地
  • 西安碑林博物馆票价将调至85元,工作人员:10元属于改扩建期间惠民票
  • 咖啡戏剧节举办第五年,上生新所“无店不咖啡,空间皆可戏”
  • 中科院院士魏辅文已卸任江西农业大学校长
  • 昆廷·斯金纳:作为“独立自主”的自由