java回调函数机制

1、 概述 java

软件模块之间老是存在着必定的接口,从调用方式上,能够把他们分为三类:同步调用、回调、异步调用 编程

同步调用:一种阻塞式调用,调用方要等待对方执行完毕才返回,它是一种单向调用; 异步

回调:一种双向调用模式,也就是说,被调用方在接口被调用时也会调用对方的接口; ide

异步调用:一种相似消息或事件的机制,解决了同步阻塞的问题,它的调用方向恰好相反,接口的服务在收到某种讯息或发生某种事件时,会主动通知客户方(即调用客户方的接口)。 函数




回调和异步调用的关系很是紧密:使用回调来实现异步消息的注册,经过异步调用来实现消息的通知。
测试


理解异步和同步 this



1.通俗说,异步就是不须要等当前执行的动做完成,就能够继续执行后面的动做。 spa

2.一般一个程序执行的顺序是:从上到下,依次执行。后面的动做必须等前面动做执行完成之后方可执行。这就是和异步相对的一个概念——同步。 .net


案例: 线程

A、张三打电话给李四,让李四帮忙写份材料。

B、李四接到电话的时候,手上有本身的工做要处理,但他答应张三,忙完手上的工做后立刻帮张三写好材料,并传真给张三。

C、通完电话后,张三外出办事。


说明:

张三给李四通完电话后,就出去办事了,他并不须要等李四把材料写好才外出。那么张三让李四写材料的消息就属于异步消息。

相反,若是张三必须等李四把材料写好才能外出办事的话,那么这个消息就属于同步消息了。


2、 异步的实现

传统的程序执行代码都是从上到下,一条一条执行。但生活中有不少状况并非这样,以上的案例中,若是李四须要几个小时之后才能帮张三写好材料的话,那张三就必须等几个小时,这样张三可能会崩溃或者抓狂。这种一条龙似的处理,显示不太合理。


可使用如下办法来处理这种问题:

张三找王五去给李四打电话,等李四写好材料后,由王五转交给张三。这样张三就能够外出办其余的事情了。

问题获得了合理的解决,以前张三一条线的工做,由张三和王五两条线来完成了,两边同时进行,彼此不耽误。

3、 计算机语言的实现

办法有了,如何用程序来模拟实现呢?

A、之前由一个线程来处理的工做,能够经过新增一个线程来达到异步的目的。

B、最后李四写好的材料必须交给张三,以作他用。这就是回调。

回调你能够这样来理解:

A发送消息给B,

B处理好A要求的事情后,将结果返回给A,

A再对B返回的结果来作进一步的处理。

4、 模拟异步消息的发送与回调

A、 回调的实现


[java] view plaincopyprint?

  1. /**

  2. * 回调接口

  3. */    

  4. public interface CallBack {    

  5.    /**

  6.     * 执行回调方法

  7.     * @param objects   将处理后的结果做为参数返回给回调方法

  8.     */    

  9.    public void execute(Object... objects );    

  10. }    


Java是面向对象的语言,所以回调函数就变成了回调接口。



B、 消息的发送者

[java] view plaincopyprint?

  1. /**

  2. * 简单本地发送异步消息的类

  3. */    

  4. public class Local implements CallBack,Runnable{    

  5.        


  6.    /**

  7.     * 远程接收消息的类,模拟point-to-point

  8.     */    

  9.    private Remote remote;    

  10.        


  11.    /**

  12.     * 发送出去的消息

  13.     */    

  14.    private String message;    

  15.        


  16.    public Local(Remote remote, String message) {    

  17.        super();    

  18.        this.remote = remote;    

  19.        this.message = message;    

  20.    }    

  21.    


  22.    /**

  23.     * 发送消息

  24.     */    

  25.    public void sendMessage()    

  26.    {    

  27.        /**当前线程的名称**/    

  28.        System.out.println(Thread.currentThread().getName());    

  29.        /**建立一个新的线程发送消息**/    

  30.        Thread thread = new Thread(this);    

  31.        thread.start();    

  32.        /**当前线程继续执行**/    

  33.        System.out.println("Message has been sent by Local~!");    

  34.    }    

  35.    


  36.    /**

  37.     * 发送消息后的回调函数

  38.     */    

  39.    public void execute(Object... objects ) {    

  40.        /**打印返回的消息**/    

  41.        System.out.println(objects[0]);    

  42.        /**打印发送消息的线程名称**/    

  43.        System.out.println(Thread.currentThread().getName());    

  44.        /**中断发送消息的线程**/    

  45.        Thread.interrupted();    

  46.    }    

  47.        


  48.    public static void main(String[] args)    

  49.    {    

  50.        Local local = new Local(new Remote(),"Hello");    

  51.            


  52.        local.sendMessage();    

  53.    }    

  54.    


  55.    public void run() {    

  56.        remote.executeMessage(message, this);    

  57.            


  58.    }    

  59. }    


C、 远程消息的接收者



[java] view plaincopyprint?

  1. /**

  2. * 处理消息的远程类

  3. *

  4. */    

  5. public class Remote {    

  6.    


  7.    /**

  8.     * 处理消息

  9.     * @param msg   接收的消息

  10.     * @param callBack  回调函数处理类

  11.     */    

  12.    public void executeMessage(String msg,CallBack callBack)    

  13.    {    

  14.        /**模拟远程类正在处理其余事情,可能须要花费许多时间**/    

  15.        for(int i=0;i<1000000000;i++)    

  16.        {    

  17.                


  18.        }    

  19.        /**处理完其余事情,如今来处理消息**/    

  20.        System.out.println(msg);    

  21.        System.out.println("I hava executed the message by Local");    

  22.        /**执行回调**/    

  23.        callBack.execute(new String[]{"Nice to meet you~!"});    

  24.    }    

  25.        


  26. }    


执行 Local 类的 main 方法。

注意Local类中:


remote.executeMessage(message, this);

executeMessage 方法须要接收一个message参数,表示发送出去的消息,而CallBack参数是他本身,也就是这里的this。表示发送消息后,由Local类本身来处理,调用自身的execute方法来处理消息结果。

若是这里不是用this,而是用其余的CallBack接口的实现类的话,那就不能称之为“回调”了,在OO的世界里,那就属于“委派”。也就是说,“回调” 必须是消息的发送者来处理消息结果,不然不能称之为回调。这个概念必须明确。



5、更多异步 + 回调 编程模式的例子


1.某天,我打电话向你请教问题,固然是个难题,你一时想不出解决方法,我又不能拿着电话在那里傻等,因而咱们约定:等你想
出办法后打手机通知我,这样,我就挂掉电话办其它事情去了。过了XX分钟,个人手机响了,你兴高采烈的说问题已经搞定,应该如此这般处理。故事到此结束。其中,你后来打手机告诉我结果即是一个“回调”过程;个人手机号码必须在之前告诉你,这即是注册回调函数;个人手机号码应该有效而且手机可以接收到你的呼叫,这是回调函数必须符合接口规范


2.有一位老板很忙,他没有时间盯着员工干活,而后他告诉本身的雇员,干完当前这些事情后,告诉他干活的结果。

建立一个回调接口,让老板得告知干完活如何找到他的方式:留下老板办公室地址:


[java] view plaincopyprint?

  1. package net.easyway.test;    

  2.    


  3. /**

  4. * 此接口为联系的方式,不管是电话号码仍是联系地址,做为

  5. * 老板都必需要实现此接口

  6. *

  7. */    

  8. public interface CallBackInterface {    

  9.    


  10.    public void execute();    

  11. }  


建立回调对象,就是老板本人,由于员工干完活后要给他打电话,所以老板必须实现回调接口,否则员工去哪里找老板?





[java] view plaincopyprint?

  1. package net.easyway.test;    

  2.    


  3. /**

  4. * 老板是做为上层应用身份出现的,下层应用(员工)是不知道

  5. * 有哪些方法,所以他想被下层应用(员工)调用必须实现此接口

  6. *

  7. */    

  8. public class Boss implements CallBackInterface {    

  9.        


  10.    @Override    

  11.    public void execute() {    

  12.        System.out.println("收到了!!" + System.currentTimeMillis());    

  13.            


  14.    }    

  15. }  


建立控制类,也就是员工对象,他必须持有老板的地址(回调接口),即便老板换了一茬又一茬,办公室不变,总能找到对应的老板。





[java] view plaincopyprint?

  1. package net.easyway.test;    

  2.    


  3. /**

  4. * 员工类,必需要记住,这是一个底层类,底层是不了解上层服务的

  5. *

  6. */    

  7. public class Employee {    

  8.    


  9.    private CallBackInterface callBack = null;    

  10.        


  11.    //告诉老板的联系方式,也就是注册    

  12.    public void setCallBack(CallBackInterface callBack){    

  13.        this.callBack = callBack;    

  14.    }    

  15.        


  16.    //工人干活    

  17.    public void doSome(){    

  18.        //1.开始干活了    

  19.        for(int i=0;i<10;i++){    

  20.            System.out.println("第【" + i + "】事情干完了!");    

  21.        }    

  22.            


  23.        //2.告诉老板干完了    

  24.        callBack.execute();    

  25.    }    

  26. }  




测试类:


[java] view plaincopyprint?

  1. package net.easyway.test;    

  2.    


  3. public class Client {    

  4.    


  5.    public static void main(String[] args) {    

  6.            


  7.            


  8.        Employee emp = new Employee();    

  9.            


  10.        //将回调对象(上层对象)传入,注册    

  11.        emp.setCallBack(new Boss());    

  12.            


  13.        //开启控制器对象运行    

  14.        emp.doSome();    

  15.    }    

  16.    


  17. }  

相关文章
相关标签/搜索