Java面向对象之接口和抽象类的区别一目了然

2024-02-28 19:12

本文主要是介绍Java面向对象之接口和抽象类的区别一目了然,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

image

介绍

相信对于Java面向对象部分,很多人很长一段时间对于接口和抽象类的区别,使用场景都不是很熟悉,同是作为抽象层重要的对象,工作中到底什么情况下使用抽象类,不是很清楚。本文就一次性把这些概念一次性说清楚,不用再烦恼了,哈哈!

核心概念

  1. 接口与抽象类最明显的区别可能就是使用上的惯用方式。接口的典型使用是代表一个类的类型或一个形容词,如 Runnable 或 Serializable,而抽象类通常是类层次结构的一部分或一件事物的类型,如 String 或 ActionHero。

  2. java8开始增加默认方法的极具说服力的理由是它允许在不破坏已使用接口的代码的情况下,在接口中增加新的方法。默认方法有时也被称为守卫方法或虚拟扩展方法。

  3. 抽象类仍然是一个类,在创建新类时只能继承它一个。而创建类的过程中可以实现多个接口。

  4. 有一条实际经验:尽可能地使用接口而不是抽象类。只有当必要时才使用抽象类。除非必须使用,否则不要用接口和抽象类。大多数时候,普通类已经做得很好,如果不行的话,再移动到接口或抽象类中。

  5. 任何抽象性都应该是由真正的需求驱动的。当有必要时才应该使用接口进行重构,而不是到处添加额外的间接层,从而带来额外的复杂性。恰当的原则是优先使用类而不是接口。从类开始,如果使用接口的必要性变得很明确,那么就重构。接口是一个伟大的工具,但它们容易被滥用。

接口和抽象类的区别

接口和抽象类都是Java中定义行为的方式,但它们之间存在一些重要的区别。

image

  1. 定义与实现

接口:接口是一种完全抽象的类型,它只包含抽象方法和常量。接口不能被实例化,只能被类实现。一个类可以实现多个接口。抽象类:抽象类是一个不完全的类,它可以包含抽象方法和非抽象方法。抽象类不能被直接实例化,需要通过子类来继承并实现所有抽象方法。一个类只能继承一个抽象类。

  1. 方法与变量

接口:接口中只能定义抽象方法(从Java 8开始也可以定义默认方法和静态方法),所有方法默认都是public的。接口中定义的变量默认是public static final的。抽象类:抽象类中可以定义普通方法、抽象方法、静态方法、构造方法等,方法默认是public或protected的。抽象类中的变量可以是任何访问修饰符。

  1. 继承与实现

接口:一个类可以实现多个接口,通过关键字implements。抽象类:一个类只能继承一个抽象类,通过关键字extends

代码示例

接口

package com.demo.java.test.javacore;interface CanFight {void fight();
}interface CanSwim {void swim();
}interface CanFly {void fly();
}class ActionCharacter {public void fight2(){System.out.println("ActionCharacter fighting");}
}class Hero extends ActionCharacter implements CanFight, CanSwim, CanFly {public void swim() {System.out.println("swiming");}public void fly() {System.out.println("flying");}@Overridepublic void fight() {System.out.println("fighting");}
}public class Adventure {public static void t(CanFight x) {x.fight();}public static void u(CanSwim x) {x.swim();}public static void v(CanFly x) {x.fly();}public static void w(ActionCharacter x) {x.fight2();}public static void main(String[] args) {Hero h = new Hero();t(h); // Treat it as a CanFightu(h); // Treat it as a CanSwimv(h); // Treat it as a CanFlyw(h); // Treat it as an ActionCharacter}
}

  • 输出:

fighting
swiming
flying
ActionCharacter fighting

抽象类

以乐器类抽象举例,请看继承关系图

image

package com.demo.java.test.javacore;
// 音符枚举
enum Note{MIDDLE_C
}/***  乐器*/
abstract class Instrument {private int i; // Storage allocated for eachpublic abstract void play(Note n);public String what() {return "Instrument";}public abstract void adjust();
}/***  管乐器*/
class Wind extends Instrument {@Overridepublic void play(Note n) {System.out.println("Wind.play() " + n);}@Overridepublic String what() {return "Wind";}@Overridepublic void adjust() {System.out.println("Adjusting Wind");}
}/***  打击乐器*/
class Percussion extends Instrument {@Overridepublic void play(Note n) {System.out.println("Percussion.play() " + n);}@Overridepublic String what() {return "Percussion";}@Overridepublic void adjust() {System.out.println("Adjusting Percussion");}
}/***  有弦乐器*/
class Stringed extends Instrument {@Overridepublic void play(Note n) {System.out.println("Stringed.play() " + n);}@Overridepublic String what() {return "Stringed";}@Overridepublic void adjust() {System.out.println("Adjusting Stringed");}
}/***  铜管乐器*/
class Brass extends Wind {@Overridepublic void play(Note n) {System.out.println("Brass.play() " + n);}@Overridepublic void adjust() {System.out.println("Adjusting Brass");}
}class Woodwind extends Wind {@Overridepublic void play(Note n) {System.out.println("Woodwind.play() " + n);}@Overridepublic String what() {return "Woodwind";}
}/***  音乐组合类*/
public class Music4 {static void tune(Instrument i) {i.play(Note.MIDDLE_C);}static void tuneAll(Instrument[] e) {for (Instrument i: e) {tune(i);}}public static void main(String[] args) {// 向上转型Instrument[] orchestra = {new Wind(),new Percussion(),new Stringed(),new Brass(),new Woodwind()};// 演凑tuneAll(orchestra);}
}

  • 输出:

Wind.play() MIDDLE_C
Percussion.play() MIDDLE_C
Stringed.play() MIDDLE_C
Brass.play() MIDDLE_C
Woodwind.play() MIDDLE_C

文章转载自:一锤子技术员

原文链接:https://www.cnblogs.com/wikiman/p/18036728

体验地址:引迈 - JNPF快速开发平台_低代码开发平台_零代码开发平台_流程设计器_表单引擎_工作流引擎_软件架构

这篇关于Java面向对象之接口和抽象类的区别一目了然的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/756366

相关文章

基于SpringBoot+Mybatis实现Mysql分表

《基于SpringBoot+Mybatis实现Mysql分表》这篇文章主要为大家详细介绍了基于SpringBoot+Mybatis实现Mysql分表的相关知识,文中的示例代码讲解详细,感兴趣的小伙伴可... 目录基本思路定义注解创建ThreadLocal创建拦截器业务处理基本思路1.根据创建时间字段按年进

Java编译生成多个.class文件的原理和作用

《Java编译生成多个.class文件的原理和作用》作为一名经验丰富的开发者,在Java项目中执行编译后,可能会发现一个.java源文件有时会产生多个.class文件,从技术实现层面详细剖析这一现象... 目录一、内部类机制与.class文件生成成员内部类(常规内部类)局部内部类(方法内部类)匿名内部类二、

SpringBoot实现数据库读写分离的3种方法小结

《SpringBoot实现数据库读写分离的3种方法小结》为了提高系统的读写性能和可用性,读写分离是一种经典的数据库架构模式,在SpringBoot应用中,有多种方式可以实现数据库读写分离,本文将介绍三... 目录一、数据库读写分离概述二、方案一:基于AbstractRoutingDataSource实现动态

Springboot @Autowired和@Resource的区别解析

《Springboot@Autowired和@Resource的区别解析》@Resource是JDK提供的注解,只是Spring在实现上提供了这个注解的功能支持,本文给大家介绍Springboot@... 目录【一】定义【1】@Autowired【2】@Resource【二】区别【1】包含的属性不同【2】@

springboot循环依赖问题案例代码及解决办法

《springboot循环依赖问题案例代码及解决办法》在SpringBoot中,如果两个或多个Bean之间存在循环依赖(即BeanA依赖BeanB,而BeanB又依赖BeanA),会导致Spring的... 目录1. 什么是循环依赖?2. 循环依赖的场景案例3. 解决循环依赖的常见方法方法 1:使用 @La

Java枚举类实现Key-Value映射的多种实现方式

《Java枚举类实现Key-Value映射的多种实现方式》在Java开发中,枚举(Enum)是一种特殊的类,本文将详细介绍Java枚举类实现key-value映射的多种方式,有需要的小伙伴可以根据需要... 目录前言一、基础实现方式1.1 为枚举添加属性和构造方法二、http://www.cppcns.co

Elasticsearch 在 Java 中的使用教程

《Elasticsearch在Java中的使用教程》Elasticsearch是一个分布式搜索和分析引擎,基于ApacheLucene构建,能够实现实时数据的存储、搜索、和分析,它广泛应用于全文... 目录1. Elasticsearch 简介2. 环境准备2.1 安装 Elasticsearch2.2 J

Java中的String.valueOf()和toString()方法区别小结

《Java中的String.valueOf()和toString()方法区别小结》字符串操作是开发者日常编程任务中不可或缺的一部分,转换为字符串是一种常见需求,其中最常见的就是String.value... 目录String.valueOf()方法方法定义方法实现使用示例使用场景toString()方法方法

Java中List的contains()方法的使用小结

《Java中List的contains()方法的使用小结》List的contains()方法用于检查列表中是否包含指定的元素,借助equals()方法进行判断,下面就来介绍Java中List的c... 目录详细展开1. 方法签名2. 工作原理3. 使用示例4. 注意事项总结结论:List 的 contain

Java实现文件图片的预览和下载功能

《Java实现文件图片的预览和下载功能》这篇文章主要为大家详细介绍了如何使用Java实现文件图片的预览和下载功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... Java实现文件(图片)的预览和下载 @ApiOperation("访问文件") @GetMapping("