模块间调用

在一个应用系统中,无论使用何种语言开发,必然存在模块之间的调用,调用的方式分为几种:

(1)同步调用

同步调用是最基本并且最简单的一种调用方式,类A的方法a()调用类B的方法b(),一直等待b()方法执行完毕,a()方法继续往下走。这种调用方式适用于方法b()执行时间不长的情况,因为b()方法执行时间一长或者直接阻塞的话,a()方法的余下代码是无法执行下去的,这样会造成整个流程的阻塞。

(2)异步调用

异步调用是为了解决同步调用可能出现阻塞,导致整个流程卡住而产生的一种调用方式。类A的方法方法a()通过新起线程的方式调用类B的方法b(),代码接着直接往下执行,这样无论方法b()执行时间多久,都不会阻塞住方法a()的执行。但是这种方式,由于方法a()不等待方法b()的执行完成,在方法a()需要方法b()执行结果的情况下(视具体业务而定,有些业务比如启异步线程发个微信通知、刷新一个缓存这种就没必要),必须通过一定的方式对方法b()的执行结果进行监听。在Java中,可以使用Future+Callable的方式做到这一点,具体做法可以参见我的这篇文章Java多线程21:多线程下其他组件之CyclicBarrier、Callable、Future和FutureTask。

(3)回调

最后是回调,回调的思想是:

  • 类A的a()方法调用类B的b()方法
  • 类B的b()方法执行完毕主动调用类A的callback()方法

这样一种调用方式组成了上图,也就是一种双向的调用方式。

代码示例

接下来看一下回调的代码示例,代码模拟的是这样一种场景:老师问学生问题,学生思考完毕回答老师。

首先定义一个回调接口,只有一个方法tellAnswer(int answer),即学生思考完毕告诉老师答案:

1 /**
2  * 回调接口,原文出处http://www.cnblogs.com/xrq730/p/6424471.html
3  */
4 public interface Callback {
5
6     public void tellAnswer(int answer);
7
8 }

定义一个老师对象,实现Callback接口:

 1 /**
 2  * 老师对象,原文出处http://www.cnblogs.com/xrq730/p/6424471.html
 3  */
 4 public class Teacher implements Callback {
 5
 6     private Student student;
 7
 8     public Teacher(Student student) {
 9         this.student = student;
10     }
11
12     public void askQuestion() {
13         student.resolveQuestion(this);
14     }
15
16     @Override
17     public void tellAnswer(int answer) {
18         System.out.println("知道了,你的答案是" + answer);
19     }
20
21 }

老师对象有两个public方法:

(1)回调接口tellAnswer(int answer),即学生回答完毕问题之后,老师要做的事情

(2)问问题方法askQuestion(),即向学生问问题

接着定义一个学生接口,学生当然是解决问题,但是接收一个Callback参数,这样学生就知道解决完毕问题向谁报告:

1 /**
2  * 学生接口,原文出处http://www.cnblogs.com/xrq730/p/6424471.html
3  */
4 public interface Student {
5
6     public void resolveQuestion(Callback callback);
7
8 }

最后定义一个具体的学生叫Ricky:

 1 /**
 2  * 一个名叫Ricky的同学解决老师提出的问题,原文出处http://www.cnblogs.com/xrq730/p/6424471.html
 3  */
 4 public class Ricky implements Student {
 5
 6     @Override
 7     public void resolveQuestion(Callback callback) {
 8         // 模拟解决问题
 9         try {
10             Thread.sleep(3000);
11         } catch (InterruptedException e) {
12
13         }
14
15         // 回调,告诉老师作业写了多久
16         callback.tellAnswer(3);
17     }
18
19 }

在解决完毕问题之后,第16行向老师报告答案。

写一个测试类,比较简单:

 1 /**
 2  * 回调测试,原文出处http://www.cnblogs.com/xrq730/p/6424471.html
 3  */
 4 public class CallbackTest {
 5
 6     @Test
 7     public void testCallback() {
 8         Student student = new Ricky();
 9         Teacher teacher = new Teacher(student);
10
11         teacher.askQuestion();
12
13     }
14
15 }

代码运行结果就一行:

知道了,你的答案是3

简单总结、分析一下这个例子就是:

(1)老师调用学生接口的方法resolveQuestion,向学生提问

(2)学生解决完毕问题之后调用老师的回调方法tellAnswer

这样一套流程,构成了一种双向调用的关系。

代码分析

分析一下上面的代码,上面的代码我这里做了两层的抽象:

(1)将老师进行抽象

  • 将老师进行抽象之后,对于学生来说,就不需要关心到底是哪位老师询问我问题,只要我根据询问的问题,得出答案,然后告诉提问的老师就可以了,即使老师换了一茬又一茬,对我学生而言都是没有任何影响的

(2)将学生进行抽象

  • 将学生进行抽象之后,对于老师这边来说就非常灵活,因为老师未必对一个学生进行提问,可能同时对Ricky、Jack、Lucy三个学生进行提问,这样就可以将成员变量Student改为List<Student>,这样在提问的时候遍历Student列表进行提问,然后得到每个学生的回答即可

这个例子是一个典型的体现接口作用的例子,之所以这么说是因为我想到有些朋友可能不太明白接口的好处,不太明白接口好处的朋友可以重点看一下这个例子,多多理解。

总结起来,回调的核心就是回调方将本身即this传递给调用方,这样调用方就可以在调用完毕之后告诉回调方它想要知道的信息。回调是一种思想、是一种机制,至于具体如何实现,如何通过代码将回调实现得优雅、实现得可扩展性比较高,一看开发者的个人水平,二看开发者对业务的理解程度。

同步回调与异步回调

上面的例子,可能有人会提出这样的疑问:

这个例子需要用什么回调啊,使用同步调用的方式,学生对象回答完毕问题之后直接把回答的答案返回给老师对象不就好了?

这个问题的提出没有任何问题,可以从两个角度去理解这个问题。

首先,老师不仅仅想要得到学生的答案怎么办?可能这个老师是个更喜欢听学生解题思路的老师,在得到学生的答案之前,老师更想先知道学生姓名和学生的解题思路,当然有些人可以说,那我可以定义一个对象,里面加上学生的姓名和解题思路不就好了。这个说法在我看来有两个问题:

(1)如果老师想要的数据越来越多,那么返回的对象得越来越大,而使用回调则可以进行数据分离,将一批数据放在回调方法中进行处理,至于哪些数据依具体业务而定,如果需要增加返回参数,直接在回调方法中增加即可

(2)无法解决老师希望得到学生姓名、学生解题思路先于学生回答的答案的问题

因此我认为简单的返回某个结果确实没有必要使用回调而可以直接使用同步调用,但是如果有多种数据需要处理且数据有主次之分,使用回调会是一种更加合适的选择,优先处理的数据放在回调方法中先处理掉。

另外一个理解的角度则更加重要,就是标题说的同步回调和异步回调了。例子是一个同步回调的例子,意思是老师向Ricky问问题,Ricky给出答案,老师问下一个同学,得到答案之后继续问下一个同学,这是一种正常的场景,但是如果我把场景改一下:

老师并不想One-By-One这样提问,而是同时向Ricky、Mike、Lucy、Bruce、Kate五位同学提问,让同学们自己思考,哪位同学思考好了就直接告诉老师答案即可。

这种场景相当于是说,同学思考完毕完毕问题要有一个办法告诉老师,有两个解决方案:

(1)使用Future+Callable的方式,等待异步线程执行结果,这相当于就是同步调用的一种变种,因为其本质还是方法返回一个结果,即学生的回答

(2)使用异步回调,同学回答完毕问题,调用回调接口方法告诉老师答案即可。由于老师对象被抽象成了Callback接口,因此这种做法的扩展性非常好,就像之前说的,即使老师换了换了一茬又一茬,对于同学来说,只关心的是调用Callback接口回传必要的信息即可

转载于:https://www.cnblogs.com/xrq730/p/6424471.html

Java回调机制解读相关推荐

  1. java回调机制及其实现(转)

    1. 什么是回调函数 回调函数,顾名思义,用于回调的函数.回调函数只是一个功能片段,由用户按照回调函数调用约定来实现的一个函数.回调函数是一个工作流的一部分,由工作流来决定函数的调用(回调)时机.回调 ...

  2. 深入浅出Java回调机制

    前几天看了一下Spring的部分源码,发现回调机制被大量使用,觉得有必要把Java回调机制的理解归纳总结一下,以方便在研究类似于Spring源码这样的代码时能更加得心应手. 注:本文不想扯很多拗口的话 ...

  3. Java反射机制解读

    Java反射机制解读 1.什么是反射 2.Hello,Java反射 3.Java程序运行的三个阶段 4.反射相关类 5.反射的优化 6.反射Class类分析 7.获取Class对象的六种方式 8.类加 ...

  4. Java回调机制是什么意思?

    华清远见成都中心 2017-09-20 10:39 一.Java回调机制是什么意思 Java中的回调机制是什么意思呢?所谓回调:就是A类中调用B类中的某个方法C,然后B类中反过来调用A类中的方法D,D ...

  5. 深入浅出: Java回调机制(异步)

    什么是回调?今天傻傻地截了张图问了下,然后被陈大牛回答道"就一个回调-".此时千万个草泥马飞奔而过(逃 哈哈,看着源码,享受着这种回调在代码上的作用,真是美哉.不妨总结总结. 一. ...

  6. Java回调机制解析

    模块之间总是存在这一定的接口,从调用方式上看,可以分为三类:同步调用.回调和异步调用.同步调用是一种阻塞式调用,也是我们在写程序中经常使用的:回调是一种双向的调用模式,也就是说,被调用的接口被调用时也 ...

  7. android 等待回调再返回结果_震惊!!只剩你还不懂Java回调机制了!

    点击上方 Java项目学习 ,选择 星标 公众号 重磅资讯.干货,第一时间送达 来源:22j.co/cFPf 调用和回调机制 在一个应用系统中, 无论使用何种语言开发, 必然存在模块之间的调用, 调用 ...

  8. 浅谈Java回调机制

    像许多网上介绍回调机制的文章一样,我这里也以一个现实的例子开头:假设你公司的总经理出差前需要你帮他办件事情,这件事情你需要花些时间去做,这时候总经理肯定不能守着你做完再出差吧, 于是就他告诉你他的手机 ...

  9. 以小故事的形式讲述Java回调机制,非常有助于理解。

    Java的回调机制,可能都听说过,当然这个机制不仅限于Java,其他各门语言,都应该有使用.但是怎么理解,就有点困惑. 还有什么是抽象类回调,什么是接口回调,以及二者之间的差别.也不是一句两句就能说清 ...

最新文章

  1. Eclipse中git检出、更新、提交、合并分支、以及解决冲突
  2. 连接到localhost后提示要求用户名和密码
  3. 读文件 —— 读写配置文件
  4. 前端Yslow的23个优化原则
  5. 《SAS编程与数据挖掘商业案例》学习笔记之三
  6. 神经网络与深度学习——TensorFlow2.0实战(笔记)(二)(安装TensorFlow2.0)
  7. 名片大全:30款精美的企业名片设计欣赏
  8. Linux系统管理第六周作业【Linux微职位】
  9. XML DOM学习笔记(JS)
  10. 使用Ajax时常用的转码方法encodeURI,escape,encodeURI
  11. Section 2.2
  12. 【吐血整理】Java项目源码分享
  13. 基于FPGA VHDL 的 ASK调制与解调设计(附源码)
  14. 痛苦的evo安装之旅,终于成功了
  15. 上海理工大计算机学研究生怎么样,上海理工大学(专业学位)计算机技术考研难吗...
  16. 我的世界java1.15.2光影_我的世界Java版带光影
  17. 12级软件测试课程博客汇总
  18. 最近网上比较火的虎年西游记金钱豹头像制作小程序源码
  19. 甲骨文oracle兴学,甲骨文 Oracle Bone Inscription
  20. Linux Signal 信号查询

热门文章

  1. node 安装_VUE项目迁移之node.js的安装
  2. 计算机基础应用形考5access答案,计算机应用基础形考作业五答案.doc
  3. 为什么说php单线程,php单线程的缺点是什么?
  4. python自动批量发邮件脚本_Python实现自动发送邮件功能
  5. idea springboot 发布webservice 发布服务_阿里云发布 Spring Boot 新脚手架,真香
  6. docker项目部署 php_「Docker部署PHP+Vue项目」- 海风纷飞Blog
  7. css初始化_利用CSS变量实现炫酷的悬浮效果
  8. C++11新特性之智能指针
  9. 【渝粤教育】电大中专会计电算化 (2)作业 题库
  10. 【渝粤题库】国家开放大学2021春1334纳税筹划题目