- 我是一个线程(修订版)
- 我是一个 Java class
- Javascript:一个屌丝的逆袭
- Java : 一个帝国的诞生
- JSP 一个装配工的没落
- TCP/IP 之 大明王朝邮差
- TCP/IP 之大明内阁
- TCP/IP 之蓟辽督师
- CPU 阿甘
- CPU 阿甘之烦恼
- CPU 阿甘:函数调用的秘密
- 我是一个网卡
- 我是一个路由器
- 我是一个进程
- 我是一块硬盘(上)
- 我是一块硬盘(下)
- 我是一个键盘
- 张大胖的 socket
- 张大胖学递归
- 学习面向对象的令狐冲
- 张大胖学数据库
- 数据库村的旺财和小强
- 小李的数据库之旅(上)
- 小李的数据库之旅(下)
- 漫画:什么是机器学习?
- 那些烦人的同步和互斥问题
- IE 为什么把火狐和 Chrome 给打伤了?
- 对浏览器村的第二次采访
- 节约标兵 IE 的自述
- EMail 诞生记
- Email 诞生记(下)
- Http 历险记(上)
- Http 历险记(下)-- Struts 的秘密
- 动物王国的面向对象
- 冯·诺伊曼计算机的诞生
- Http Server : 一个差生的逆袭
- 张大胖的加法器
- 从 1 加到 100:一道简单的数学题挑战下你的大脑
- 编程语言
- Javascript:一个屌丝的逆袭
- 计算机语言之战
- 我和编程语言的爱恨情仇(上)
- 我和编程语言的爱恨情仇(下)
- Android 为什么选择了 Java
- iOS 为什么选择了 Object-C?
- Basic : 一个老兵的自述
- Node.js : 我只需要一个店小二
- 命令式编程 vs 声明式编程
- 编译还是解释?
- 程序人生
- “架构师"小赵
- 师兄说
- 师姐说
- 小王的架构师之路
- 小李的版本管理系统
- 小超穿越记
- 小李的 Build 之路(上)
- 小李的 Build 之路(下)
- 张大胖改 Bug
- 我的编程之路--大学趣事
- 码农小王的一天
- 小李在外企
- 张大胖的需求估算
- 从厨师到码农
- 聊一聊那些神一样的程序员们(上)
- 聊一聊那些神一样的程序员们(中)
- 聊一聊那些神一样的程序员们(下)
- 谁是互联网之父?
- 一个价值百万的创业教训
- 让自己与众不同 - 提升工作的价值
- 看看你的“易燃性”
- 从无聊的工作中寻找价值
- 什么样的学生适合报考计算机?
- 谈谈程序员的职业方向(上)
- 谈谈程序员的职业方向(中)
- 谈谈程序员的职业方向(下)
- 谈谈培训班的作用
- 码农需要知道的“潜规则”
- 学习编程的加速度
- 码农在工作中的必备能力
- 码农和英语
- 老司机经验
- 假如时光能够倒流, 我会这么学习 Java
- 假如我是计算机系老师
- 学会编程, 而不是学会 Java
- 从增删改查中突围
- 抽象:程序员必备的能力
- 懒就一个字
- 编程的自学方法
- 小王买房记
- 从一道面试题谈谈一线码农应该具备的基本素质
- 想写框架的看过来
- 苹果手机变砖头以后
- 如何快速的学习一门技术?
- 唯一不变的是变化: 谈谈微信应用号
- 什么是企业应用?
- 勿以浮沙筑高台
- 为什么敏捷开发难于成功?
- localhost vs 127.0.0.1
- GitHub/Stackoverflow 找工作时有什么用?
- 动词 or 名词 :这是一个问题
- 如何选择入行语言
- 有时候,沉默是金
- 零 Bug 的代码是怎么炼成的?
- 浮点数为什么不精确?
- 文章错误大全
- Open Source--不要为了开源而开源
- 一不留神,代码就腐化了
- 先做个“键盘侠”, 再来写程序
- 不加断点调试的程序员是好程序员
- 码农必备技能:烂代码的处理之道(上)
- 码农必备技能:烂代码的处理之道(下)
- 学习数据结构有用吗?
- 从现在开始,丰富你的简历
- 那些永不过时的书,你看过几本吗?
- 学好编程必备的一个品质你知道吗?
- 你最爱的 Java
- 搞懂了这几点,你就学会了 Web 编程
- Spring 的本质系列(1) -- 依赖注入
- Spring 本质系列(2)-AOP
- 三层架构和 MVC 那点事儿
- Java 帝国之拨云见日识回调
- 小张的 Duck Typing
- JDBC 的诞生
- JDBC 后传
- 一个不安分的 JDBC 驱动
- Java 帝国之 Java bean (上)
- Java 帝国之 Java bean(下)
- Java 帝国之函数式编程
- Java 帝国之函数式编程(下)
- 关于 Java 初学者需要知道的 10 件事
- JUnit 你不知道的那些事儿
- 圣诞礼物:Java EE 的历史
- Java EE 读书指南
- 给小白的 Java EE 指南
- 给小白的 Java EE 指南(2)
- 给小白的 Java EE 生存指南(3) : XML
- 给小白的 Java EE 生存指南(4) : 一只叫 Tom 的猫
- 给小白的 Java EE 指南(5) : AJAX
- 给小白的 Java EE 生存指南(6) :Java 反射
- 闲聊
- "饿了么"初体验
- 来自大脑的控诉
- 一个高中生是怎么玩自媒体的?
- 尝试 分答
- 到底应不应该上培训班?
- 自学编程中遇到问题怎么办?
- 据说 99%的初级程序员看完后都不迷茫了
- 一行代码引发的“血案”
- 对一个死锁问题的思考
- 通过外包进入名企
- 请开往十年前的今天
- 为什么自学中最好有个师傅指导一下?
- 这个网站值得你花时间投入
- 为什么你无法坚持自学编程?
Java 帝国之拨云见日识回调
前言:本文原作者是 王钦誉, 原文链接: https://xiaoqinyu0000.github.io/Java/JavaCallback/
我对文章做了修改, 文章主要介绍 Java 的回调机制及其实现。
故事背景
在日常编程中,我们经常需要对内存的数据进行持久化的工作,把他们保存在硬盘文件或者数据库中。
为了避免重复, 我们通常会把这部分工作封装在一个工具类中, 让各个客户端来调用。
下文的 FileIO 就是一个简单的工具类(为了简单起见,并没有使用单例或静态方法来实现)
小张的烦恼
Java 帝国的 FileIO 是一个忙碌的家伙,附近 7、8 个村落的人都来找他, 请他把数据存储到硬盘里。
FileIO 提供了一个简单的接口, 大家只要告诉他文件名和要保存的字符串内容, 剩下的事就只是等待了,FileIO 会完成工作,告诉大家是成功还是失败。
比如张家村的小张来请 FileIO 保存文件的时候是这样的:
通常情况下, 小张都会很快拿到返回结果, 高高兴兴的回家。
但这一次不知道怎么回事, 这个 FileIO 一直不返回结果, 把小张阻塞了长达 1 秒钟!
小张说: “哥们, 怎么回事? 我这儿都等了 1000 毫秒了, 还没完?
FileIO 回答 : ”这不能怪我啊, 你这次的数据量实在是太大了,是谁上传的大文件故意捣乱吧, 对了, 你杀毒没有?“
"安全问题不用你考虑 " 小张也有点底气不足 :”我觉得数据量还行, 也有可能是硬盘这会儿太忙了”
总之,小张一直阻塞在那里,无法回家。
回调
阻塞的事情发生的多了,极大的影响了小张的工作, 最近这一周的工分可是落后了不少啊, 再这么下去,月底分粮的时候就要饿肚子了, 饿肚子还是小事, 自己喜欢的张二妮看到没粮食,估计就不理我了。。。
他拎了两瓶好酒去找 FileIO 商量: “兄弟, 我听说有一种异步保存的办法, 你那边能不能用下? 保存数据的时候起一个线程, 把主线程让回给我,保存好了再通知我,我也不用老是等你,是吧?”
FIleIO 想了想说:“这样确实可以解决问题,但每天找我保存数据的人也很多,而且我也不知道在完成数据的写入之后怎么通知你呢?”
小张把两瓶好酒往前一推, “我们关系这么好,你再开个专属我的方法呗,我在调用你的 saveStrToFile 方法的时候顺便把我的实例给你,你搞完之后通过我的实例调用我的方法通知我就行啦。就调我的 onResult() 这个方法吧。这事要保密, 天知地知你知我知就行了”。
于是,FileIO 为小张开了一个 VIP 通道:
这种方式很巧妙,小张调用 FileIO 的 saveStrToFile(String,String,XiaoZhang) 的时候,把自己的实例通过第三个参数给了 FileIO,FileIO 开启子线程保存完数据之后,通过 XiaoZhang 给的实例回调 onResult(boolean) 方法。
听起来很绕口,但总结起来就 我调你的方法,你再回调我的方法 。
后来,JAVA 帝国给这种机制取了个名字叫 回调机制 ,在帝国中广为人知。
酒后泄密
由于有了 FileIO 的 VIP 通道,小张处理业务的能力大幅度提升, 工分不但在张家村独占鳌头, 就是算上李家村, 刘家村 等,那也是数一数二的。
小张一时风光不已,越来越多的人来向他请教秘诀,但小张却笑而不语(这可是成功秘诀,能告诉你们嘛...)。
有一次, 李家村的小李看到了 FileIO 有了一个新接口(毕竟都是公开的嘛), 但是不知道怎么回事, 自己也调用不了, 类型不对啊。
小李别有用心的请小张和 FileIO 喝酒, 酒过三巡, 俩人终于吐露了这个秘密。
这一下子炸开了锅, 虽然 Java 帝国规定, 接口的设计一定要规范, 不能乱来, 但是大家蜂拥而至, 纷纷要求 FileIO 给自己也开 VIP 通道。
FileIO 实在是没有办法, 无奈之下为小李, 小王等等都开启了 VIP 通道:
村长支招
随着 FileIO 开启的 VIP 通道越来越多,FIleIO 发现自己的体积越来越膨胀,自己有大量的代码是在处理这些 VIP 通道,而且处理方式都差不多,VIP 通道多了也就失去其意义了。
有一次, 张家村德高望重的村长路过 FileIO 这里,FileIO 知道村长软件设计能力了得, 赶紧拉住就行讨教。
村长果然见(lao) 多(jian) 识(ju) 广(hua),“小伙子,既然我们村的成员老是需要你的帮助,你就别为每个人开启一个 VIP 通道,你直接弄一个我们张家村的 VIP 通道,这个通道不是接受张大胖, 张二胖这样的类, 而是接受一个 ZhangClient 的抽象类。这个抽象类中只有一个方法:onResult
每次,有人去找你帮忙的时候,你也不用管具体是谁,只要他实现了 ZhangClient,你就知道它有一个 onResult(false)的方法,你处理完了之后直接回调它的 onResult(boolean) 方法就行了,是不是很简单啊,哈哈哈哈哈~~~”
FileIO 听完老村长的话恍然大悟, 这一层解决不了的事,那我们再加一层,在上一层解决呗 :
如上所示,FileIO 表面上回调了 ZhangClient 的 onResult(false) 方法,但实际上回调的是 XiaoZhang 的 onResult(false) 方法,因为传进来的实例实际上是继承了 ZhangClient 的小张(作者:感觉像披着羊皮的狼)。
后来,帝国将这种利用抽象类去实现回调的方式称之为 抽象类回调 。
Java 巡视组
FileIO 把其他通道都删除了, 只留了一个 ZhangClient 通道, 现在他明白老村长的老奸巨猾了。
因为李家村、赵家村、王家村的人都抱怨说, 我们找你保存个数据, 还得继承一个姓 Zhang 的类, 实在是太扯了!
FileIO 想了想, 得了, 为了避免引起众怒, 还是改个名称吧,就叫 FileIOClient 。
即使是这样, 很多人还在抱怨: 我已经继承了一个类了, 怎么可能再继承你这个 FileIOClient ? 不继承就没法保存数据, 还有没有王法了! 还有,你这老是改来该去, 把我们都该累死了。
事情闹大了, 上面派了个巡视组下来解决。
FileIO 战战兢兢的给巡视组诉苦: ”我也实在是没办法啊, 你看 Java 也不允许多继承, 我昨晚想起一个办法, JAVA 类都隐性继承 Object,能不能在 Object 里面增加一个回调的方法?“
巡视组生气的说:”别做梦了! java.lang.Object 是我们的根, 那是你加方法的地方吗?! 你整天只知道保存数据, 难道都忘了 Java 帝国的接口(interface)了吗?“
FileIO 被点醒了, 既然继承的方式搞不定,那就接口好了, 接口可以随意实现, 想实现几个实现几个。
在巡查组的监视下, FileIO 很快修改了代码:
不幸的是, 大家的代码也都得改一遍, 万幸的是, 只需要把 extends FileIOClient 改为 implements IFileIOCallBack 即可。
后来,帝国将这种利用接口去实现回调机制的方式称之为 接口回调。
尾声
张家村的小张有点落寞, 他原来独有的回调方法现在已经被接口回调所替代,他独有的优势已经荡然无存,风光不再。
更让他烦心的事, 随着 FileIO 接口的变化, 他的代码也不断的改来改去, 光是修改就耽误了不少事儿,少挣了好多工分。
不就是一个回调吗, 还继承这个, 实现那个的, 这 Java 搞的也太复杂了。
有小道消息说,Java 帝国之外的动态语言王国有个叫 Duck Typing 的东西, 实现回调的时候根本不用继承什么东西, 也不用实现什么接口, 只要自己有一个 onResult 方法, 就可以被调用, 小张好奇心大起,决定去出去闯一闯。
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论