我的编程空间,编程开发者的网络收藏夹
学习永远不晚

Java中的回调机制怎么实现

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

Java中的回调机制怎么实现

本篇内容介绍了“Java中的回调机制怎么实现”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!

模块间的调用

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

(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的函数体里通过书写函数B的函数名来调用之,使内存中对应函数B的代码得以执行。异步调用是一种类似消息或事件的机制解决了同步阻塞的问题,例如 A通知 B后,他们各走各的路,互不影响,不用像同步调用那样, A通知 B后,非得等到 B走完后, A才继续走 。回调是一种双向的调用模式,也就是说,被调用的接口被调用时也会调用对方的接口,例如A要调用B,B在执行完又要调用A。

回调的用途
回调一般用于层间协作,上层将本层函数安装在下层,这个函数就是回调,而下层在一定条件下触发回调。例如作为一个驱动,是一个底层,他在收到一个数据时,除了完成本层的处理工作外,还将进行回调,将这个数据交给上层应用层来做进一步处理,这在分层的数据通信中很普遍。

多线程中的“回调”

Java多线程中可以通过callable和future或futuretask结合来获取线程执行后的返回值。实现方法是通过get方法来调用callable的call方法获取返回值。

其实这种方法本质上不是回调,回调要求的是任务完成以后被调用者主动回调调用者的接口。而这里是调用者主动使用get方法阻塞获取返回值。

public class 多线程中的回调 {    //这里简单地使用future和callable实现了线程执行完后    public static void main(String[] args) throws ExecutionException, InterruptedException {        ExecutorService executor = Executors.newCachedThreadPool();        Future<String> future = executor.submit(new Callable<String>() {            @Override            public String call() throws Exception {                System.out.println("call");                TimeUnit.SECONDS.sleep(1);                return "str";            }        });        //手动阻塞调用get通过call方法获得返回值。        System.out.println(future.get());        //需要手动关闭,不然线程池的线程会继续执行。        executor.shutdown();    //使用futuretask同时作为线程执行单元和数据请求单元。    FutureTask<Integer> futureTask = new FutureTask(new Callable<Integer>() {        @Override        public Integer call() throws Exception {            System.out.println("dasds");            return new Random().nextInt();        }    });    new Thread(futureTask).start();    //阻塞获取返回值    System.out.println(futureTask.get());}@Testpublic void test () {    Callable callable = new Callable() {        @Override        public Object call() throws Exception {            return null;        }    };    FutureTask futureTask = new FutureTask(callable);}}

Java回调机制实战

曾经自己偶尔听说过回调机制,隐隐约约能够懂一些意思,但是当让自己写一个简单的示例程序时,自己就傻眼了。随着工作经验的增加,自己经常听到这儿使用了回调,那儿使用了回调,自己是时候好好研究一下Java回调机制了。网上关于Java回调的文章一抓一大把,但是看完总是云里雾里,不知所云,特别是看到抓取别人的代码走两步时,总是现眼。于是自己决定写一篇关于Java机制的文章,以方便大家和自己更深入的学习Java回调机制。

首先,什么是回调函数,引用百度百科的解释:回调函数就是一个通过函数指针调用的函数。如果你把函数的指针(地址)作为参数传递给另一个函数,当这个指针被用来调用其所指向的函数时,我们就说这是回调函数。回调函数不是由该函数的实现方直接调用,而是在特定的事件或条件发生时由另外的一方调用的,用于对该事件或条件进行响应[2].

不好意思,上述解释我看了好几遍,也没理解其中深刻奥秘,相信一些读者你也一样。光说不练假把式,咱们还是以实战理解脉络。

实例一 : 同步调用

本文以底层服务BottomService和上层服务UpperService为示例,利用上层服务调用底层服务,整体执行过程如下:

执行UpperService.callBottomService();

执行BottomService.bottom();

执行UpperService.upperTaskAfterCallBottomService()

1.1 同步调用代码

同步调用时序图:

Java中的回调机制怎么实现

同步调用时序图

1.1.1 底层服务类:BottomService.java

package synchronization.demo;public class BottomService {public String bottom(String param) {try { //  模拟底层处理耗时,上层服务需要等待Thread.sleep(3000);} catch (InterruptedException e) {e.printStackTrace();}return param +" BottomService.bottom() execute -->";}}

1.1.2 上层服务接口: UpperService.java

package synchronization.demo;public interface UpperService {public void upperTaskAfterCallBottomService(String upperParam);public String callBottomService(final String param);}

1.1.3 上层服务接口实现类:UpperServiceImpl.java

package synchronization.demo;public class UpperServiceImpl implements UpperService {private BottomService bottomService;@Overridepublic void upperTaskAfterCallBottomService(String upperParam) {System.out.println(upperParam + " upperTaskAfterCallBottomService() execute.");}public UpperServiceImpl(BottomService bottomService) {this.bottomService = bottomService;}@Overridepublic String callBottomService(final String param) {return bottomService.bottom(param + " callBottomService.bottom() execute --> ");}}

1.1.4 Test测试类:Test.java

package synchronization.demo;import java.util.Date;public class Test {public static void main(String[] args) {BottomService bottomService = new BottomService();UpperService upperService = new UpperServiceImpl(bottomService);System.out.println("=============== callBottomService start ==================:" + new Date());String result = upperService.callBottomService("callBottomService start --> ");//upperTaskAfterCallBottomService执行必须等待callBottomService()调用BottomService.bottom()方法返回后才能够执行upperService.upperTaskAfterCallBottomService(result);System.out.println("=============== callBottomService end ====================:" + new Date());}}

1.1.5 输出结果:

=============== callBottomService start ==================:Thu Jan 19 14:59:58 CST 2017callBottomService start -->  callBottomService.bottom() execute -->  BottomService.bottom() execute --> upperTaskAfterCallBottomService() execute.=============== callBottomService end ====================:Thu Jan 19 15:00:01 CST 2017

注意输出结果:

是同步方式,Test调用callBottomService()等待执行结束,然后再执行下一步,即执行结束。callBottomService开始执行时间为Thu Jan 19 14:59:58 CST 2017,执行结束时间为Thu Jan 19 15:00:01 CST 2017,耗时3秒钟,与模拟的耗时时间一致,即3000毫秒。

实例二:由浅入深

前几天公司面试有问道java回调的问题,因为这方面也没有太多研究,所以回答的含糊不清,这回特意来补习一下。看了看网上的回调解释和例子,都那么的绕口,得看半天才能绕回来,其实吧,回调是个很简单的机制。在这里我用简单的语言先来解释一下:假设有两个类,分别是A和B,在A中有一个方法a(),B中有一个方法b();在A里面调用B中的方法b(),而方法b()中调用了方法a(),这样子就同时实现了b()和a()两个方法的功能。

疑惑:为啥这么麻烦,我直接在类A中的B.b()方法下调用a()方法就行了呗。
解答:回调更像是一个约定,就是如果我调用了b()方法,那么就必须要回调,而不需要显示调用
一、Java的回调-浅
我们用例子来解释:小明和小李相约一起去吃早饭,但是小李起的有点晚要先洗漱,等小李洗漱完成后,通知小明再一起去吃饭。小明就是类A,小李就是类B。一起去吃饭这个事件就是方法a(),小李去洗漱就是方法b()。

public class XiaoMing {       //小明和小李一起吃饭   public void eatFood() {      XiaoLi xl = new XiaoLi();      //A调用B的方法      xl.washFace();   }   public void eat() {      System.out.print("小明和小李一起去吃大龙虾");   }}那么怎么让小李洗漱完后在通知小明一起去吃饭呢public class XiaoMing {       //小明和小李一起吃饭   public void eatFood() {      XiaoLi xl = new XiaoLi();      //A调用B的方法      xl.washFace();      eat();   }   public void eat() {      System.out.print("小明和小李一起去吃大龙虾");   }}

不过上面已经说过了这个不是回调函数,所以不能这样子,正确的方式如下

public class XiaoLi{//小李   public void washFace() {    System.out.print("小李要洗漱");    XiaoMing xm = new XiaoMing();        //B调用A的方法    xm.eat();//洗漱完后,一起去吃饭   }}

这样子就可以实现washFace()同时也能实现eat()。小李洗漱完后,再通知小明一起去吃饭,这就是回调。

二、Java的回调-中
可是细心的伙伴可能会发现,小李的代码完全写死了,这样子的场合可能适用和小明一起去吃饭,可是假如小李洗漱完不吃饭了,想和小王上网去,这样子就不适用了。其实上面是伪代码,仅仅是帮助大家理解的,真正情况下是需要利用接口来设置回调的。现在我们继续用小明和小李去吃饭的例子来讲讲接口是如何使用的。

小明和小李相约一起去吃早饭,但是小李起的有点晚要先洗漱,等小李洗漱完成后,通知小明再一起去吃饭。小明就是类A,小李就是类B。不同的是我们新建一个吃饭的接口EatRice,接口中有个抽象方法eat()。在小明中调用这个接口,并实现eat();小李声明这个接口对象,并且调用这个接口的抽象方法。这里可能有点绕口,不过没关系,看看例子就很清楚了。

EatRice接口:

public interface EatRice {   public void eat(String food);}小明:public class XiaoMing implements EatRice{//小明   //小明和小李一起吃饭   public void eatFood() {    XiaoLi xl = new XiaoLi();    //A调用B的方法    xl.washFace("大龙虾", this);//this指的是小明这个类实现的EatRice接口   }   @Override   public void eat(String food) {    // TODO Auto-generated method stub    System.out.println("小明和小李一起去吃" + food);   }}小李:public class XiaoLi{//小李   public void washFace(String food,EatRice er) {    System.out.println("小李要洗漱");        //B调用了A的方法    er.eat(food);   }}测试Demo:public class demo {   public static void main(String args[]) {    XiaoMing xm = new XiaoMing();    xm.eatFood();   }}

测试结果:

这样子就通过接口的形式实现了软编码。通过接口的形式我可以实现小李洗漱完后,和小王一起去上网。代码如下

public class XiaoWang implements EatRice{//小王   //小王和小李一起去上网   public void eatFood() {    XiaoLi xl = new XiaoLi();    //A调用B的方法    xl.washFace("轻舞飞扬上网", this);   }   @Override   public void eat(String bar) {    // TODO Auto-generated method stub    System.out.println("小王和小李一起去" + bar);   }}

实例三:Tom做题

数学老师让Tom做一道题,并且Tom做题期间数学老师不用盯着Tom,而是在玩手机,等Tom把题目做完后再把答案告诉老师。

1 数学老师需要Tom的一个引用,然后才能将题目发给Tom。

2 数学老师需要提供一个方法以便Tom做完题目以后能够将答案告诉他。

3 Tom需要数学老师的一个引用,以便Tom把答案给这位老师,而不是隔壁的体育老师。

回调接口,可以理解为老师接口

    //回调指的是A调用B来做一件事,B做完以后将结果告诉给A,这期间A可以做别的事情。    //这个接口中有一个方法,意为B做完题目后告诉A时使用的方法。    //所以我们必须提供这个接口以便让B来回调。    //回调接口,    public interface CallBack {        void tellAnswer(int res);    }

数学老师类

    //老师类实例化回调接口,即学生写完题目之后通过老师的提供的方法进行回调。    //那么学生如何调用到老师的方法呢,只要在学生类的方法中传入老师的引用即可。    //而老师需要指定学生答题,所以也要传入学生的实例。public class Teacher implements CallBack{    private Student student;    Teacher(Student student) {        this.student = student;    }    void askProblem (Student student, Teacher teacher) {        //main方法是主线程运行,为了实现异步回调,这里开启一个线程来操作        new Thread(new Runnable() {            @Override            public void run() {                student.resolveProblem(teacher);            }        }).start();        //老师让学生做题以后,等待学生回答的这段时间,可以做别的事,比如玩手机.\        //而不需要同步等待,这就是回调的好处。        //当然你可以说开启一个线程让学生做题就行了,但是这样无法让学生通知老师。        //需要另外的机制去实现通知过程。        // 当然,多线程中的future和callable也可以实现数据获取的功能。        for (int i = 1;i < 4;i ++) {            System.out.println("等学生回答问题的时候老师玩了 " + i + "秒的手机");        }    }    @Override    public void tellAnswer(int res) {        System.out.println("the answer is " + res);    }}

学生接口

    //学生的接口,解决问题的方法中要传入老师的引用,否则无法完成对具体实例的回调。    //写为接口的好处就是,很多个学生都可以实现这个接口,并且老师在提问题时可以通过    //传入List<Student>来聚合学生,十分方便。public interface Student {    void resolveProblem (Teacher teacher);}

学生Tom

public class Tom implements Student{    @Override    public void resolveProblem(Teacher teacher) {        try {            //学生思考了3秒后得到了答案,通过老师提供的回调方法告诉老师。            Thread.sleep(3000);            System.out.println("work out");            teacher.tellAnswer(111);        } catch (InterruptedException e) {            e.printStackTrace();        }    }

测试类

public class Test {    public static void main(String[] args) {        //测试        Student tom = new Tom();        Teacher lee = new Teacher(tom);        lee.askProblem(tom, lee);        //结果//        等学生回答问题的时候老师玩了 1秒的手机//        等学生回答问题的时候老师玩了 2秒的手机//        等学生回答问题的时候老师玩了 3秒的手机//        work out//        the answer is 111    }}

“Java中的回调机制怎么实现”的内容就介绍到这里了,感谢大家的阅读。如果想了解更多行业相关的知识可以关注编程网网站,小编将为大家输出更多高质量的实用文章!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

Java中的回调机制怎么实现

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

Java中的回调机制怎么实现

本篇内容介绍了“Java中的回调机制怎么实现”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!模块间的调用在一个应用系统中,无论使用何种语言开发
2023-06-02

java回调机制实例详解

java回调机制实例详解以前不理解什么叫回调,天天听人家说加一个回调方法啥的,心里想我草,什么叫回调方法啊?然后自己就在网上找啊找啊找,找了很多也不是很明白,现在知道了,所谓回调:就是A类中调用B类中的某个方法C,然后B类中反过来调用A类中
2023-05-31

什么是Java回调机制

这篇文章主要讲解了“什么是Java回调机制”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“什么是Java回调机制”吧!一、回调回调分为同步回调和异步回调, 假如以买彩票的场景来模拟, 我买彩票
2023-06-20

怎么在java项目中运用回调机制

怎么在java项目中运用回调机制?很多新手对此不是很清楚,为了帮助大家解决这个难题,下面小编将为大家详细讲解,有这方面需求的人可以来学习下,希望你能有所收获。代码如下:public class Student { private Str
2023-05-31

Java中怎么实现SPI机制

Java中怎么实现SPI机制,很多新手对此不是很清楚,为了帮助大家解决这个难题,下面小编将为大家详细讲解,有这方面需求的人可以来学习下,希望你能有所收获。2 什么是SPI机制SPI是Service Provider Interface 的简
2023-06-16

python中的垃圾回收机制怎么实现

这篇文章主要讲解了“python中的垃圾回收机制怎么实现”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“python中的垃圾回收机制怎么实现”吧!python采用的是引用计数机制为主,标记-清
2023-06-28

怎么用Java实现回调例程

小编给大家分享一下怎么用Java实现回调例程,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!熟悉 MS-windows 和 X Window System 事件驱动
2023-06-03

JAVA中怎么实现反射机制

这期内容当中小编将会给大家带来有关JAVA中怎么实现反射机制,文章内容丰富且以专业的角度为大家分析和叙述,阅读完这篇文章希望大家可以有所收获。JAVA反射机制定义:  JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属
2023-06-17

Java实现回调的写法

本篇内容介绍了“Java实现回调的写法”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!1. 引言在计算机程序设计中,回调函数,简称回调(Cal
2023-06-16

Java中的synchronized锁膨胀机制怎么实现

这篇文章主要讲解了“Java中的synchronized锁膨胀机制怎么实现”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Java中的synchronized锁膨胀机制怎么实现”吧!synch
2023-06-30

C++中怎么实现回调函数

今天就跟大家聊聊有关C++中怎么实现回调函数,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。C++回调函数代码示例:#include < string> #include < iost
2023-06-17

ASP.NET中怎么实现异步回调

ASP.NET中怎么实现异步回调,相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。ASP.NET异步回调的重要性:在实际开发中,ASP.NET 客户端异步回调技术才是王道。下面我们
2023-06-17

Java项目中是如何实现回调的

这篇文章将为大家详细讲解有关Java项目中是如何实现回调的,文章内容质量较高,因此小编分享给大家做个参考,希望大家阅读完这篇文章后对相关知识有一定的了解。回调回调的概念:举个例子就是,我们想要问别人一道题,我们把题跟对方说了一下,对方说好,
2023-05-31

java超时机制怎么实现

在Java中,可以使用java.util.Timer和java.util.concurrent.Executors等类来实现超时机制。java.util.Timer:创建一个定时器,使用schedule方法来安排超时操作。可以使用Timer
java超时机制怎么实现
2024-02-29

JAVA中怎么实现反射机制功能

本篇文章为大家展示了JAVA中怎么实现反射机制功能,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。JAVA基础--JAVA反射机制详解 JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这
2023-06-17

Java中怎么实现日志缓存机制

今天就跟大家聊聊有关Java中怎么实现日志缓存机制,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。Java 日志机制的介绍Java 日志机制在很多文章中都有介绍,为了便于后面文章部分的
2023-06-17

怎么在PHP中实现一个垃圾回收机制

这篇文章给大家介绍怎么在PHP中实现一个垃圾回收机制,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。顽固垃圾的产生过程代码中,$a变量内部存储信息为a: (refco
2023-06-06

在Java中如何实现回调过程

这篇文章主要介绍了在Java中如何实现回调过程,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。在Java中实现回调过程XML:namespace prefix = o ns =
2023-06-03

java怎么实现Redis的LRU缓存机制

本篇内容主要讲解“java怎么实现Redis的LRU缓存机制”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“java怎么实现Redis的LRU缓存机制”吧!目录LRU概述使用LinkedHashM
2023-06-20

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录