`
HelloSure
  • 浏览: 311144 次
  • 性别: Icon_minigender_1
  • 来自: 武汉
社区版块
存档分类
最新评论

深入浅出Java回调机制

阅读更多
前几天看了一下Spring的部分源码,发现回调机制被大量使用,觉得有必要把Java回调机制的理解归纳总结一下,以方便在研究类似于Spring源码这样的代码时能更加得心应手。

注:本文不想扯很多拗口的话来充场面,我的目的是希望以最简明扼要的语言将Java回调的大概机制说清楚。好了,言归正传。

一句话,回调是一种双向调用模式,什么意思呢,就是说,被调用方在被调用时也会调用对方,这就叫回调。“If you call me, i will call back”。
不理解?没关系,先看看这个可以说比较经典的使用回调的方式
  • class A实现接口InA ——背景1
  • class A中包含一个class B的引用b ——背景2
  • class B有一个参数为InA的方法test(InA a) ——背景3
  • A的对象a调用B的方法传入自己,test(a) ——这一步相当于you call me
  • 然后b就可以在test方法中调用InA的方法 ——这一步相当于i call you back

是不是清晰一点了?下面再来看一个完全符合这个方式模板的例子
(PS:这个例子来源于网络,由于这个例子表现的功能极度拉风,令我感觉想想出一个超越它的例子确实比较困难,所以直接搬过来)
//相当于接口InA
public interface BoomWTC{
  //获得拉登的决定
  public benLaDengDecide();

  // 执行轰炸世贸
  public void boom();
}

//相当于class A
public class At$911 implements BoomWTC{//相当于【背景1】
  private boolean decide;
  private TerroristAttack ta;//相当于【背景2】

  public At$911(){
    Date now=new Date();
    SimpleDateFormat myFmt1=new SimpleDateFormat("yy/MM/dd HH:mm");
    this.dicede= myFmt.format(dt).equals("01/09/11 09:44");
    this.ta=new TerroristAttack();
  }

  //获得拉登的决定
  public boolean benLaDengDecide(){
    return decide;
  }

  // 执行轰炸世贸
  public void boom(){
    ta.attack(new At$911);//class A调用class B的方法传入自己的对象,相当于【you call me】
  }
}

//相当于class B
public class TerroristAttack{
  public TerroristAttack(){
  }

  public attack(BoomWTC bmw){——这相当于【背景3】
    if(bmw.benLaDengDecide()){//class B在方法中回调class A的方法,相当于【i call you back】
     //let's go.........
    }
  }
}

现在应该对回调有一点概念了吧。
可是问题来了,对于上面这个例子来说,看不出用回调有什么好处,直接在调用方法不就可以了,为什么要使用回调呢
事实上,很多需要进行回调的操作是比较费时的,被调用者进行费时操作,然后操作完之后将结果回调给调用者。看这样一个例子:
//模拟Spring中HibernateTemplate回调机制的代码
    interface CallBack{   
        public void doCRUD();   
    }  
      
    public class HibernateTemplate {   
          
        public void execute(CallBack action){  
            getConnection();  
            action.doCRUD();  
            releaseConnection();  
        }  
       
        public void add(){  
             execute(new CallBack(){  
                public void doCRUD(){  
                    System.out.println("执行add操作...");  
                }  
             });  
        }   
      
        public void getConnection(){  
            System.out.println("获得连接...");  
        }  
          
        public void releaseConnection(){  
            System.out.println("释放连接...");  
        }  
          
    }  

可能上面这个例子你不能一眼看出个所以然来,因为其实这里A是作为一个内部匿名类存在的。好,不要急,让我们把这个例子来重构一下:
interface CallBack{   //相当于接口InA
    public void doCRUD();   
}  

public class A implements CallBack{//【背景1】
    private B b;//【背景2】
    public void doCRUD(){  
          System.out.println("执行add操作...");  
     }  

     public void add(){  
             b.execute(new A());//【you call me】  
        }  
}

public class B{
     public void execute(CallBack action){  //【背景3】
            getConnection();  
            action.doCRUD();  //【i call you back】
            releaseConnection();  
        }  

      public void getConnection(){  
            System.out.println("获得连接...");  
        }  
          
        public void releaseConnection(){  
            System.out.println("释放连接...");  
        }  
}

好了,现在就明白多了吧,完全可以转化为上面所说的回调使用方式的模板。
现在在来看看为什么要使用回调,取得连接getConnection();是费时操作,A希望由B来进行这个费时的操作,执行完了之后通知A即可(即所谓的i call you back)。这就是这里使用回调的原因。

在网上看到了一个比喻,觉得很形象,这里借用一下:
你有一个复杂的问题解决不了,打电话给你的同学,你的同学说可以解决这个问题,但是需要一些时间,那么你不可能一直拿着电话在那里等,你会把你的电话号码告诉他,让他解决之后打电话通知你。回调就是体现在你的同学又反过来拨打你的号码。
结合到前面所分析的,你打电话给你同学就是【you call me】,你同学解决完之后打电话给你就是【i call you back】。

怎么样,现在理解了吧? 

---------------------------------以下为更新----------------------------------

看了有些朋友的回帖,我又思考了一下,感觉自己之前对回调作用的理解的确存在偏差。
下面把自己整理之后的想法共享一下,如果有错误希望指出!多谢!

先说上面这段代码,本来完全可以用模板模式来进行实现:
public abstract class B{
     public void execute(){ 
            getConnection();  
            doCRUD();  
            releaseConnection();  
        }  

      public abstract void doCRUD();

      public void getConnection(){  
            System.out.println("获得连接...");  
        }  
          
        public void releaseConnection(){  
            System.out.println("释放连接...");  
        }  
}

public class A extends B{
    public void doCRUD(){  
          System.out.println("执行add操作...");  
     }  

     public void add(){  
             doCRUD();
        }  
}

public class C extends B{
    public void doCRUD(){  
          System.out.println("执行delete操作...");  
     }  

     public void delete(){  
             doCRUD();
        }  
}

如果改为回调实现是这样的:
    interface CallBack{   
        public void doCRUD();   
    }  
      
    public class HibernateTemplate {   
        public void execute(CallBack action){  
            getConnection();  
            action.doCRUD();  
            releaseConnection();  
        }  
       
        public void add(){  
             execute(new CallBack(){  
                public void doCRUD(){  
                    System.out.println("执行add操作...");  
                }  
             });  
         }   

         public void delete(){  
             execute(new CallBack(){  
                public void doCRUD(){  
                    System.out.println("执行delete操作...");  
                }  
             });  
         } 
      
        public void getConnection(){  
            System.out.println("获得连接...");  
        }  
          
        public void releaseConnection(){  
            System.out.println("释放连接...");  
        }  
          
    }  

可见摒弃了继承抽象类方式的回调方式更加简便灵活。不需要为了实现抽象方法而总是继承抽象类,而是只需要通过回调来增加一个方法即可,更加的直观简洁灵活。这算是回调的好处之一。

下面再给出一个关于利用回调配合异步调用的很不错的例子,来源于http://kt8668.iteye.com/blog/205739
回调接口:
public interface CallBack {  
    /** 
     * 执行回调方法 
     * @param objects   将处理后的结果作为参数返回给回调方法 
     */  
    public void execute(Object... objects );  
}  

消息的发送者:
/**
 * 这个类相当于你自己
 */
public class Local implements CallBack,Runnable{  
   
    private Remote remote;  
      
    /** 
     * 发送出去的消息 
     */  
    private String message;  
      
    public Local(Remote remote, String message) {  
        super();  
        this.remote = remote;  
        this.message = message;  
    }  
  
    /** 
     * 发送消息 
     */  
    public void sendMessage()  
    {  
        /**当前线程的名称**/  
        System.out.println(Thread.currentThread().getName());  
        /**创建一个新的线程发送消息**/  
        Thread thread = new Thread(this);  
        thread.start();  
        /**当前线程继续执行**/  
        System.out.println("Message has been sent by Local~!");  
    }  
  
    /** 
     * 发送消息后的回调函数 
     */  
    public void execute(Object... objects ) {  
        /**打印返回的消息**/  
        System.out.println(objects[0]);  
        /**打印发送消息的线程名称**/  
        System.out.println(Thread.currentThread().getName());  
        /**中断发送消息的线程**/  
        Thread.interrupted();  
    }  
      
    public static void main(String[] args)  
    {  
        Local local = new Local(new Remote(),"Hello");  
          
        local.sendMessage();  
    }  
  
    public void run() {  
        remote.executeMessage(message, this);  //这相当于给同学打电话,打完电话之后,这个线程就可以去做其他事情了,只不过等到你的同学打回电话给你的时候你要做出响应
          
    }  
}  

消息的接收者:
/**
 * 这个类相当于你的同学
 */
public class Remote {  
  
    /** 
     * 处理消息 
     * @param msg   接收的消息 
     * @param callBack  回调函数处理类 
     */  
    public void executeMessage(String msg,CallBack callBack)  
    {  
        /**模拟远程类正在处理其他事情,可能需要花费许多时间**/  
        for(int i=0;i<1000000000;i++)  
        {  
              
        }  
        /**处理完其他事情,现在来处理消息**/  
        System.out.println(msg);  
        System.out.println("I hava executed the message by Local");  
        /**执行回调**/  
        callBack.execute(new String[]{"Nice to meet you~!"});  //这相当于同学执行完之后打电话给你
    }  
      
}  

由上面这个例子可见,回调可以作为异步调用的基础来实现异步调用。
36
16
分享到:
评论
3 楼 energykey 2011-07-22  
支持一下。
建议列出更多适用的场景,或者结合spring的源码给出适用场景。
当然,你的例子足以支撑本文的主题。
if...better...
2 楼 huqiji 2011-07-22  
不错,确实是好东西,分享万岁
1 楼 Jclick 2011-07-22  
很不错,赞一个!

相关推荐

    java 最简答的回调理解

    本文将深入浅出地解释Java回调的概念,并通过实例来帮助你更好地掌握这一技术。 首先,回调的基本思想是让一个函数或方法作为参数传递给另一个函数,当这个被调用的函数执行完毕后,它会调用我们传入的函数。这种...

    深入浅出JNA—快速调用原生函数

    通过JNA,开发者可以更轻松地调用原生函数,也更容易实现原生代码中回调Java方法的功能。JNA的Pointer类作为基础,模拟指针的各种操作,如指针引用、指针解引用等,为操作复杂数据结构提供了极大的便利。 举例来说...

    深入浅出Netty

    Netty对Java NIO进行了封装和抽象,简化了原生Java NIO的使用,同时通过统一的API接口和事件回调机制,使得开发者可以更加专注于业务逻辑的处理,而不用关心底层的NIO细节。 Netty的源码虽然复杂,但是其设计理念和...

    深入浅出 C#

    《深入浅出 C#》是一本旨在帮助读者深入理解C#编程语言的资源集合,它包含了一系列的练习题(exercise.txt)、阅读材料(readme.txt)以及如何学习C#的指南(howcs)。这本书的核心目标是使读者不仅能够熟练地编写C#...

    深入浅出c#基础知识中文版

    8. **事件和委托**:事件和委托是C#中的重要特性,用于实现组件间的通信和回调机制,常见于用户界面编程。 9. **泛型接口和委托**:结合泛型和接口,可以创建强大的设计模式,比如工厂模式和策略模式。委托则可用于...

    深入浅出Android

    ### 深入浅出Android:全面解析与学习指南 #### 一、Android简介与历史背景 Android是一款基于Linux内核的操作系统,由Andy Rubin等人于2003年创立的Android公司研发,并在2005年被谷歌收购。自2008年起,随着第一...

    深入浅出JNA—快速调用原生函数1.0.pdf.zip_jna

    标准映射用于调用原生函数,而回调映射则允许原生代码调用Java方法,实现双向通信。 总的来说,JNA提供了一种灵活且易于使用的机制,让Java开发者可以轻松地与操作系统底层交互,而无需深入理解JNI的细节。然而,...

    Netty 高并发深入浅出学习高并发服务器

    Netty 是一个高性能、异步事件驱动的网络应用程序框架,专为 Java 平台...《深入浅出Netty.pdf》这本书很可能会详细讲解这些知识点,包括理论基础、实践示例以及高级特性的使用,对于理解和掌握 Netty 有着极大的帮助。

    深入浅出Android--Google手持设备应用程序设计

    《深入浅出Android--Google手持设备应用程序设计》是一本针对Android开发的专业指南,旨在帮助读者理解和掌握在Google手持设备上创建应用程序的全过程。这本书详细介绍了Android操作系统的核心特性、开发环境的搭建...

    ext+js深入浅出

    ### ext+js深入浅出:关键技术知识点解析 #### 一、EXTJS概述 **EXTJS**,简称**EXT**,是一种先进的Ajax框架,用于构建视觉效果丰富的客户端应用程序。该框架完全由JavaScript编写而成,能够独立于后端技术,这...

    浅谈Java内部类的四个应用场景

    无论是提供私有访问控制、简化事件监听器的实现、提高代码复用性还是实现回调机制,内部类都是一个非常实用且强大的工具。希望本文能够帮助读者更深入地理解Java内部类的应用,并在实际开发中灵活运用。

    深入浅出android开发PDF

    开发者需要理解这些状态的变化,并在相应的生命周期回调方法中处理应用数据的保存和恢复。 数据存储是Android应用开发中常见的需求。Android提供了多种数据存储方式,比如文件存储、SharedPreferences、数据库存储...

    c#技术c#技术c#技术c#技术

    5. **委托与事件**:委托是C#中的类型,代表方法的引用,常用于实现回调机制和事件处理。事件是发布者和订阅者之间通信的方式。 6. **泛型**:泛型允许创建可重用的类型,这些类型可以操作多种数据类型,提高了代码...

Global site tag (gtag.js) - Google Analytics