自从给小白写了两篇科普性质的文章后,我就有点一发不可收拾,觉得很有必要继续写下去。因为有读者留言“鼓励”我说,“二哥,你真的是为小白操碎了心啊!”我容易吗?我。

当我们要完成的任务是确定的,但具体的方式需要随后开个会投票的话,Java 的抽象类就派上用场了。这句话怎么理解呢?搬个小板凳坐好,听我来给你讲讲。

01、抽象类的 5 个关键点

1)定义抽象类的时候需要用到关键字 abstract,放在 class 关键字前。

public abstract class AbstractPlayer {}

关于抽象类的命名,阿里出品的 Java 开发手册上有强调,“抽象类命名要使用 Abstract 或 Base 开头”,记住了哦。

2)抽象类不能被实例化,但可以有子类。

尝试通过 new 关键字实例化的话,编译器会报错,提示“类是抽象的,不能实例化”。

通过 extends 关键字可以继承抽象类,继承后,BasketballPlayer 类就是 AbstractPlayer 的子类。

public class BasketballPlayer extends AbstractPlayer {}

3)如果一个类定义了一个或多个抽象方法,那么这个类必须是抽象类。

当在一个普通类(没有使用 abstract 关键字修饰)中定义了抽象方法,编译器就会有两处错误提示。

第一处在类级别上,提醒你“这个类必须通过 abstract 关键字定义”,or 的那个信息没必要,见下图。

第二处在方法级别上,提醒你“抽象方法所在的类不是抽象的”,见下图。

4)抽象类可以同时声明抽象方法和具体方法,也可以什么方法都没有,但没必要。就像下面这样:

public abstract class AbstractPlayer {abstract void play();public void sleep() {System.out.println("运动员也要休息而不是挑战极限");}
}

5)抽象类派生的子类必须实现父类中定义的抽象方法。比如说,抽象类中定义了 play() 方法,子类中就必须实现。

public class BasketballPlayer extends AbstractPlayer {@Overridevoid play() {System.out.println("我是张伯伦,篮球场上得过 100 分");}
}

如果没有实现的话,编译器会提醒你“子类必须实现抽象方法”,见下图。

02、什么时候用抽象类

与抽象类息息相关的还有一个概念,就是接口,我们留到下一篇文章中详细说,因为要说的知识点还是蛮多的。你现在只需要有这样一个概念就好,接口是对行为的抽象,抽象类是对整个类(包含成员变量和行为)进行抽象。

(是不是有点明白又有点不明白,别着急,翘首以盼地等下一篇文章出炉吧)

除了接口之外,还有一个概念就是具体的类,就是不通过 abstract 修饰的普通类,见下面这段代码中的定义。

public class BasketballPlayer {public void play() {System.out.println("我是詹姆斯,现役第一人");}
}

有接口,有具体类,那什么时候该使用抽象类呢?

1)我们希望一些通用的功能被多个子类复用。比如说,AbstractPlayer 抽象类中有一个普通的方法 sleep(),表明所有运动员都需要休息,那么这个方法就可以被子类复用。

public abstract class AbstractPlayer {public void sleep() {System.out.println("运动员也要休息而不是挑战极限");}
}

虽然 AbstractPlayer 类可以不是抽象类——把 abstract 修饰符去掉也能满足这种场景。但 AbstractPlayer 类可能还会有一个或者多个抽象方法。

BasketballPlayer 继承了 AbstractPlayer 类,也就拥有了 sleep() 方法。

public class BasketballPlayer extends AbstractPlayer {}

BasketballPlayer 对象可以直接调用 sleep() 方法:

BasketballPlayer basketballPlayer = new BasketballPlayer();
basketballPlayer.sleep();

FootballPlayer 继承了 AbstractPlayer 类,也就拥有了 sleep() 方法。

public class FootballPlayer extends AbstractPlayer {}

FootballPlayer 对象也可以直接调用 sleep() 方法:

FootballPlayer footballPlayer = new FootballPlayer();
footballPlayer.sleep();

2)我们需要在抽象类中定义好 API,然后在子类中扩展实现。比如说,AbstractPlayer 抽象类中有一个抽象方法 play(),定义所有运动员都可以从事某项运动,但需要对应子类去扩展实现。

public abstract class AbstractPlayer {abstract void play();
}

BasketballPlayer 继承了 AbstractPlayer 类,扩展实现了自己的 play() 方法。

public class BasketballPlayer extends AbstractPlayer {@Overridevoid play() {System.out.println("我是张伯伦,我篮球场上得过 100 分,");}
}

FootballPlayer 继承了 AbstractPlayer 类,扩展实现了自己的 play() 方法。

public class FootballPlayer extends AbstractPlayer {@Overridevoid play() {System.out.println("我是C罗,我能接住任意高度的头球");}
}

3)如果父类与子类之间的关系符合 is-a 的层次关系,就可以使用抽象类,比如说篮球运动员是运动员,足球运动员是运动员。

03、具体示例

为了进一步展示抽象类的特性,我们再来看一个具体的示例。假设现在有一个文件,里面的内容非常简单——“Hello World”,现在需要有一个读取器将内容读取出来,最好能按照大写的方式,或者小写的方式。

这时候,最好定义一个抽象类,比如说 BaseFileReader:

public abstract class BaseFileReader {protected Path filePath;protected BaseFileReader(Path filePath) {this.filePath = filePath;}public List<String> readFile() throws IOException {return Files.lines(filePath).map(this::mapFileLine).collect(Collectors.toList());}protected abstract String mapFileLine(String line);
}

filePath 为文件路径,使用 protected 修饰,表明该成员变量可以在需要时被子类访问。

readFile() 方法用来读取文件,方法体里面调用了抽象方法 mapFileLine()——需要子类扩展实现大小写的方式。

你看,BaseFileReader 设计的就非常合理,并且易于扩展,子类只需要专注于具体的大小写实现方式就可以了。

小写的方式:

public class LowercaseFileReader extends BaseFileReader {protected LowercaseFileReader(Path filePath) {super(filePath);}@Overrideprotected String mapFileLine(String line) {return line.toLowerCase();}
}

大写的方式:

public class UppercaseFileReader extends BaseFileReader {protected UppercaseFileReader(Path filePath) {super(filePath);}@Overrideprotected String mapFileLine(String line) {return line.toUpperCase();}
}

你看,从文件里面一行一行读取内容的代码被子类复用了——抽象类 BaseFileReader 类中定义的普通方法 readFile()。与此同时,子类只需要专注于自己该做的工作,LowercaseFileReader 以小写的方式读取文件内容,UppercaseFileReader 以大写的方式读取文件内容。

接下来,我们来新建一个测试类 FileReaderTest:

public class FileReaderTest {public static void main(String[] args) throws URISyntaxException, IOException {URL location = FileReaderTest.class.getClassLoader().getResource("helloworld.txt");Path path = Paths.get(location.toURI());BaseFileReader lowercaseFileReader = new LowercaseFileReader(path);BaseFileReader uppercaseFileReader = new UppercaseFileReader(path);System.out.println(lowercaseFileReader.readFile());System.out.println(uppercaseFileReader.readFile());}
}

项目的 resource 目录下有一个文本文件,名字叫 helloworld.txt。

可以通过 ClassLoader.getResource() 的方式获取到该文件的 URI 路径,然后就可以使用 LowercaseFileReader 和 UppercaseFileReader 两种方式读取到文本内容了。

输出结果如下所示:

[hello world]
[HELLO WORLD]


好了,我亲爱的读者朋友,以上就是本文的全部内容了。是不是感觉认知边界又拓宽了?

我是沉默王二,一枚有趣的程序员。如果觉得文章对你有点帮助,请微信搜索「 沉默王二 」第一时间阅读,回复【666】更有我为你精心准备的 500G 高清教学视频(已分门别类)。

本文 GitHub 已经收录,有大厂面试完整考点,欢迎 Star。

原创不易,莫要白票,请你为本文点个赞吧,这将是我写作更多优质文章的最强动力。

小白,你要的Java抽象类,操碎了心!相关推荐

  1. 蜂鸟智游大数据:“人在囧途”的春运,航空公司们可操碎了心

    "有钱没钱回家过年",这句歌词唱出了很多人的心声.临近年关,回家和家人团聚成了所有在外漂泊游子的头等大事,回家的想法是美好的,但路途却是坎坷的. "一票难求"是 ...

  2. 在高考作弊这事上,古今监考老师操碎了心!

    作者 | 神经小姐姐 本文经授权转自HyperAI超神经(ID:HyperAI) 从古至今,有考试的地方就有作弊,作弊方法可谓是五花八门,妖魔鬼怪各显神通.尽管有法律约束与层层安检搜查把关,却依然难防 ...

  3. 冒险岛当前服务器不稳定,冒险岛频繁卡顿PDD又为粉丝操碎了心,这次要出面调整服务器...

    原标题:冒险岛频繁卡顿PDD又为粉丝操碎了心,这次要出面调整服务器 说起PDD相信大家都不会陌生,在游戏领域PDD算得上一个游戏小王子了.不管是什么类型的游戏PDD都有在玩,近几日PDD又在直播间玩起 ...

  4. 为了弄个微信公众号也是操碎了心,欢迎关注”恐漫屋“

    为什么80%的码农都做不了架构师?>>>    看伊藤润二全集请关注微信公众号"恐漫屋",马上拿出微信扫一扫 很久很久以前看过祝耕夫的<黑水潭>,这应 ...

  5. 为了让你看技术文章,我们操碎了心。。。

    众所周知,大家不爱看技术文章.不管是大号还是小号,公号主们都有这么一种感受:辛辛苦苦写了一天的干货文章,阅读数寥寥无几,花半小时写的一篇水文,阅读立竿见影瞬间破万.究其原因还是技术文章烧脑,而非技术文 ...

  6. 为大家学习Flutter,阿里操碎了心

    Flutter中文网: https://flutterchina.club/ 本文转载自公众号  鸿洋 国内在实践Flutter以及对外分享方面,闲鱼可以说一直位于前列,不断输出了非常多的实践类文章, ...

  7. 郭台铭操碎了心 夏普并入富士康后怎么走

    "过程是艰辛的,结局是完美的."富士康创始人郭台铭如此形容对夏普的收购. 富士康收购夏普一案从年前一直拖到了今年4月份,中间甚至一度传出富士康要放弃,整个收购过程并不轻松.夏普并入 ...

  8. - 转载- 阿里为大家学习Flutter操碎了心

    1.背景: 这是泓洋大佬今早发的一篇文章,本人所处团队目前也在接触Flutter.Dart,毕竟是Google的亲儿子,怎么也得给点面子,学习一下. 传送门:"阿里为大家学习Flutter操 ...

  9. 人工智能为人们减肥操碎了心

    减脂行业玩家不少,但行业痛点十分明显.较为典型的是通过智能可穿戴设备"卖数据"和通过代餐"卖产品".商业模式无可厚非,"减肥"是女性永恒不变 ...

最新文章

  1. Node.js链式回调
  2. Halcon初学者知识【17】如何将零件提取dxf图
  3. linux 安装 memcached
  4. Notepad++ 设置执行 lua 和 python
  5. 当下流行的分布式文件系统大阅兵
  6. mysql group concat_MySQL 的 GROUP_CONCAT 函数详解
  7. Mvvm 前端数据流框架精讲
  8. 使用yum命令安装服务时,一直卡在Loaded plugins: fastestmirror Determining fastest mirrors
  9. java实现deflate算法
  10. 使用dime传输大附件的设置(WSE Soap toolkit)
  11. LINUX也有C#?
  12. jquery.treeview.js
  13. 《FLUENT 14流场分析自学手册》——第1章 流体力学基础 1.1 流体力学基本概念
  14. 【社会网络分析图】python实现
  15. caj转换成word的方法,每天免费使用
  16. 项目管理:精益管理法
  17. 模型组合方法-boosting算法详解(机器学习面试必备)
  18. Vue3.0实现原生高度可自定义菜单组件vue3-menus
  19. MySQL--数据库基础知识点(一)
  20. 苹果手机(iPhone)系统升级到IOS16.1后,发现连接WiFi、热点总是经常自动(随机)断开

热门文章

  1. 用Java面向对象的方法重写“吃货联盟订餐系统”
  2. 腾讯云 WebShell 体验
  3. 如何通过微调整功能,解决打印位置上下偏移的问题?(转)
  4. 如何批量修改文件夹中图片的后缀名
  5. 弘辽科技:拼多多拼单价格怎么算?拼单规则是什么?
  6. Android开机画面的具体修改方法
  7. word中插入公式编辑器
  8. 两个视频左右拼接成一个视频,且两个视频音频都保留
  9. KDD-cup 2019比赛总结
  10. 诺基亚android one机型,诺基亚7 Plus曝光,搭载Android One+骁龙660