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

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;}
}
http://www.xdnf.cn/news/4921.html

相关文章:

  • 高效C/C++之十:Coverity修复问题:尽量多使用 c++强制类型转化
  • 人工智能之数学基础:二次型
  • 内网渗透——红日靶场三
  • HOT 100 | 【子串】76.最小覆盖子串、【普通数组】53.最大子数组和、【普通数组】56.合并区间
  • AI与计算机视觉(CV):目标检测与图像分割的最新进展
  • 行业 |四大痛点待破:“拆解”DeepSeek一体机
  • 英伟达Blackwell架构重构未来:AI算力革命背后的技术逻辑与产业变革
  • 【强化学习】动态规划(Dynamic Programming, DP)算法
  • Jenkins集成Maven
  • 如何构建容器镜像并将其推送到极狐GitLab容器镜像库?
  • 【亲测有效】如何清空但不删除GitHub仓库中的所有文件(main分支)
  • 单例模式的两种设计
  • [论文阅读]Deeply-Supervised Nets
  • Idea Code Templates配置
  • K8S - Harbor 镜像仓库部署与 GitLab CI 集成实战
  • 工业与协议融合篇:如何将多个协议集成进一个系统?
  • OpenCV中适用华为昇腾(Ascend)后端的逐元素操作(Per-element Operations)
  • MCU存储系统架构解析
  • 面试问题(连载。。。。)
  • 【Bootstrap V4系列】学习入门教程之 组件-下拉菜单(Dropdowns)
  • k8s术语之secret
  • 数据库——关系代数之扩展操作
  • Java—— 集合 Collection
  • SNMP 协议介绍、开发方法及示例
  • 0X. Linux嵌入式系统(课堂笔记)
  • 基于STM32、HAL库的FT232RL USB转UART收发器 驱动程序设计
  • 基于主成分分析(PCA)的数据降维
  • 泰勒展开式
  • Ollama调优,提高性能与安全性
  • 15.thinkphp的上传功能