Java回调机制详解[通俗易懂]

Java回调机制详解[通俗易懂]网上关于Java回调的文章一抓一大把,但是看完总是云里雾里,不知所云,特别是看到抓取别人的代码走两步时,总是现眼。于是自己决定写一篇关于Java机制的文章,以方便大家和自己更深入的学习Java回调机制。

大家好,又见面了,我是你们的朋友全栈君。

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

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

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

1.实战

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

          第一步: 执行UpperService.callBottomService();

          第二步: 执行BottomService.bottom();

          第三步:执行UpperService.upperTaskAfterCallBottomService()

       1.1 同步调用代码

                         同步调用时序图:

Java回调机制详解[通俗易懂]

              底层服务类:BottomService.java                  

package synchronization.demo;

/**
 * Created by zhangzh on 2017/1/19.
 */
public class BottomService {



    public String bottom(String param) {

        try { //  模拟底层处理耗时,上层服务需要等待
            Thread.sleep(3000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }

        return param +" BottomService.bottom() execute -->";

    }

}

        上层服务接口: UpperService.java

package synchronization.demo;

/**
 * Created by zhangzh on 2017/1/19.
 */
public interface UpperService {

    public void upperTaskAfterCallBottomService(String upperParam);

    public String callBottomService(final String param);
}

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

package synchronization.demo;

/**
 * Created by zhangzh on 2017/1/19.
 */
public class UpperServiceImpl implements UpperService {

    private BottomService bottomService;

    @Override
    public void upperTaskAfterCallBottomService(String upperParam) {

        System.out.println(upperParam + " upperTaskAfterCallBottomService() execute.");
    }

    public UpperServiceImpl(BottomService bottomService) {
        this.bottomService = bottomService;
    }

    @Override
    public String callBottomService(final String param) {

        return bottomService.bottom(param + " callBottomService.bottom() execute --> ");
    }

}

      Test测试类:Test.java

package synchronization.demo;

import java.util.Date;

/**
 * Created by zhangzh on 2017/1/19.
 */
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());

    }
}

      输出结果:       

=============== callBottomService start ==================:Thu Jan 19 14:59:58 CST 2017
callBottomService 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毫秒。

1.2 Java回调实战

        同步调用很明显的缺点是UpperService必须等待BottomService的返回结果之后,才能够继续向下执行upperTaskAfterCallBottomService(),造成了UpperService必须等待。然而回调则不需要等待,回调只需要将UpperService自身实例的引用或指针传给BottomService,BottomService执行完bottom()后,通过UpperService实例的引用或指针调用upperTaskAfterCallBottomService(),达到了与同步调用同样的效果且UpperService的调用者Test不需要等待。

         回调的执行时序图如下:

Java回调机制详解[通俗易懂]

     具体实现代码如下: 

             底层服务类:BottomService.java        

package callback.demo;

/**
 * Created by zhangzh on 2017/1/19.
 */
public class BottomService {



    public void bottom(UpperService upperService, String param) {

        try {
            Thread.sleep(3000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }

        upperService.upperTaskAfterCallBottomService( param +" bottom callback upperTaskAfterCallBottomService() execute -->");

    }

}

       上层服务接口: UpperService.java      

package callback.demo;

/**
 * Created by zhangzh on 2017/1/19.
 */
public interface UpperService {

    public void upperTaskAfterCallBottomService(String upperParam);

    public void callBottomService(final String param);
}

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

package callback.demo;

/**
 * Created by zhangzh on 2017/1/19.
 */
public class UpperServiceImpl implements UpperService {

    private BottomService bottomService;

    @Override
    public void upperTaskAfterCallBottomService(String upperParam) {

        System.out.println( upperParam + " upperTaskAfterCallBottomService() execute.");
    }

    public UpperServiceImpl(BottomService bottomService) {
        this.bottomService = bottomService;
    }

    @Override
    public void callBottomService(final String param) {

        new Thread(new Runnable() {

            public void run() {
                bottomService.bottom(UpperServiceImpl.this,    param  + " callBottomService.bottom() execute --> ");
            }
        }).start();


    }


}

    Test测试类:Test.java 

package callback.demo;

import java.util.Date;

/**
 * Created by zhangzh on 2017/1/19.
 */
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());;

        upperService.callBottomService("callBottomService start --> ");

        System.out.println("=============== callBottomService end ====================:" + new Date());

    }
}

      输出结果:     

=============== callBottomService start ==================:Thu Jan 19 16:00:45 CST 2017
=============== callBottomService end ====================:Thu Jan 19 16:00:45 CST 2017
callBottomService start -->  callBottomService.bottom() execute -->  bottom callback upperTaskAfterCallBottomService() execute --> upperTaskAfterCallBottomService() execute.

         注意输出结果:

         Test不需要等待callBottomService()执行结束后才进行下一步执行,Test继续往下执行结束,不需要等待callBottomService()执行完毕。 故callBottomService的从开始到执行结束的时间都是Thu Jan 19 16:00:45 CST 2017,执行耗时时间为0秒,而执行效果达到和同步执行一致。

          反观1.1同步方式的执行耗时情况,callBottomService开始执行时间为Thu Jan 19 14:59:58 CST 2017,执行结束时间为Thu Jan 19 15:00:01 CST 2017,耗时3秒钟,与模拟的耗时时间一致,即3000毫秒。

参考资料:1.http://blog.csdn.net/xiaanming/article/details/8703708/

                    2.http://baike.baidu.com/link?url=g6KCkbYIxHECm0PCGVtq6j59yLqrNNE8MW6jP6pXY_OS2QIw43w_3YbjP86gQJozXnw0MtPBaFLhBfwh7zVs7kmMj2nXswVcqNiFaS3_zXL3JjE-alQqdeeDFpgV4JhO

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/149578.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • mysql数据库备份和还原的命令_Mysql数据库备份和还原常用的命令

    mysql数据库备份和还原的命令_Mysql数据库备份和还原常用的命令Mysql数据库备份和还原常用的命令是进行Mysql数据库备份和还原的关键,没有命令,什么都无从做起,更谈不上什么备份还原,只有给系统这个命令,让它去执行,才能完成Mysql数据库备份和还原的操作,下面就是操作的常用命令。一、备份命令1、备份MySQL数据库的命令mysqldump-hhostname-uusername-ppassworddatabasename>backupf…

    2022年5月4日
    56
  • 解决报错(Navigation cancelled from “/roleList“ to “/userlist“ with a new navigation.)[通俗易懂]

    解决报错(Navigation cancelled from “/roleList“ to “/userlist“ with a new navigation.)[通俗易懂]写项目的时候,报了一个错,现在总结出来,希望可以帮助到你们。这个报错的原因:使用新导航取消了从“/roleList”到“/userlist”的导航。解决的方法:关于VueRouter报错路由

    2022年7月1日
    155
  • 10款流程图绘制工具

    10款流程图绘制工具介绍10款免费、强大的在线流程图工具,绝对满足你对流程图的所有幻想!1.draw.iodraw.io是一款可以作为首选、强推的一款流程工具,不仅支持在线版,还可以安装到Windows、Mac进行离线使用,尤其是与VSCode的结合,让它支持的平台进一步得到丰富。2.ProcessOnProcessOn[2]可以算得上是一款老牌、知名的在线流程图工具。它不仅支持流程图,还支持思维导图、原型图、网络拓扑图、组织结构图、UML等。目前免费版支持个人文件存储9个,团队协作.

    2025年8月3日
    2
  • pycharm在全局搜索某个变量「建议收藏」

    pycharm在全局搜索某个变量「建议收藏」吃了写代码不规范的亏。。。明明记得之前写过某个功能,但是因为文件命名太随意,“123”,“111”,“a”,“test”等不知所谓的名称都被我用上了,以至于找代码时候焦头烂额,以后一定要养成规范命名的好习惯。不过如果已经这样写了,需要找某个函数时,可以用全局搜索的方式,通过找关键字,定位函数,定位文件。具体操作参看:pycharm如何全局进行查找一个关键词…

    2022年5月8日
    200
  • 腾讯云 Ubuntu16.04.1 允许 root 用户 SSH登录 修改 sshd_config PermitRootLogin 后登录失败[通俗易懂]

    腾讯云 Ubuntu16.04.1 允许 root 用户 SSH登录 修改 sshd_config PermitRootLogin 后登录失败[通俗易懂]腾讯云Ubuntu16.04.1允许root用户SSH登录修改sshd_configPermitRootLogin后登录失败今天重装了吃灰很久的服务器,选择的是Ubuntu,腾讯云默认会创建名为ubuntu的用户用于ssh登录。ubuntu执行某些命令时权限不够需要加sudo,我超级不喜欢输入命令行时加上sudo的╰(‵□′)╯。解决

    2022年5月29日
    48
  • linux 挖矿效率_以太坊(ETH)怎么挖矿?LinuxETH系统挖矿教程「建议收藏」

    linux 挖矿效率_以太坊(ETH)怎么挖矿?LinuxETH系统挖矿教程「建议收藏」Geth客户端的CPU挖矿效率非常低,纯属鸡肋。如对CPU挖矿感兴趣,请参照本文。下面介绍GPU挖矿。硬件GPU挖矿需要1-2GB显存(译者注:1GB显存不可以),显存不够会一直报错。GPU矿机是基于OpenCL的,所以AMD显卡比NVIDI显卡更有效率。ASIC和FPGA(即专业矿机)是无效的。根据自己的显卡类型和系统,选择openCL:AMDSDKopenCLNVIDIACUDAope…

    2022年10月15日
    2

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号