2015.4.26.00.14_回调机制_深入浅出java回调机制

0-深入浅出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:这个例子来源于网络,由于这个例子表现的功能极度拉风,令我感觉想想出一个超越它的例子确实比较困难,所以直接搬过来)
Java代码

1.	//相当于接口InA  
2.	public interface BoomWTC{  
3.	  //获得拉登的决定  
4.	  public benLaDengDecide();  
5.	  
6.	  // 执行轰炸世贸  
7.	  public void boom();  
8.	}  
9.	  
10.//相当于class A  
11.public class At$911 implements BoomWTC{//相当于【背景1】  
12.  private boolean decide;  
13.  private TerroristAttack ta;//相当于【背景2】  
14.  
15.  public At$911(){  
16.    Date now=new Date();  
17.    SimpleDateFormat myFmt1=new SimpleDateFormat("yy/MM/dd HH:mm");  
18.    this.dicede= myFmt.format(dt).equals("01/09/11 09:44");  
19.    this.ta=new TerroristAttack();  
20.  }  
21.  
22.  //获得拉登的决定  
23.  public boolean benLaDengDecide(){  
24.    return decide;  
25.  }  
26.  
27.  // 执行轰炸世贸  
28.  public void boom(){  
29.    ta.attack(new At$911);//class A调用class B的方法传入自己的对象,相当于【you call me】  
30.  }  
31.}  
32.  
33.//相当于class B  
34.public class TerroristAttack{  
35.  public TerroristAttack(){  
36.  }  
37.  
38.  public attack(BoomWTC bmw){——这相当于【背景3】  
39.    if(bmw.benLaDengDecide()){//class B在方法中回调class A的方法,相当于【i call you back】  
40.     //let's go.........  
41.    }  
42.  }  
43.}  

现在应该对回调有一点概念了吧。 
可是问题来了,对于上面这个例子来说,看不出用回调有什么好处,直接在调用方法不就可以了,为什么要使用回调呢? 
事实上,很多需要进行回调的操作是比较费时的,被调用者进行费时操作,然后操作完之后将结果回调给调用者。看这样一个例子: 
Java代码   
1.//模拟Spring中HibernateTemplate回调机制的代码  
2.    interface CallBack{     
3.        public void doCRUD();     
4.    }    
5.        
6.    public class HibernateTemplate {     
7.            
8.        public void execute(CallBack action){    
9.            getConnection();    
10.            action.doCRUD();    
11.            releaseConnection();    
12.        }    
13.         
14.        public void add(){    
15.             execute(new CallBack(){    
16.                public void doCRUD(){    
17.                    System.out.println("执行add操作...");    
18.                }    
19.             });    
20.        }     
21.        
22.        public void getConnection(){    
23.            System.out.println("获得连接...");    
24.        }    
25.            
26.        public void releaseConnection(){    
27.            System.out.println("释放连接...");    
28.        }    
29.            
30.    }    

可能上面这个例子你不能一眼看出个所以然来,因为其实这里A是作为一个内部匿名类存在的。好,不要急,让我们把这个例子来重构一下:
Java代码

1.	interface CallBack{   //相当于接口InA  
2.	    public void doCRUD();     
3.	}    
4.	  
5.	public class A implements CallBack{//【背景1】  
6.	    private B b;//【背景2】  
7.	    public void doCRUD(){    
8.	          System.out.println("执行add操作...");    
9.	     }    
10.	  
11.	     public void add(){    
12.	             b.execute(new A());//【you call me】    
13.	        }    
14.	}  
15.	  
16.	public class B{  
17.	     public void execute(CallBack action){  //【背景3】  
18.	            getConnection();    
19.	            action.doCRUD();  //【i call you back】  
20.	            releaseConnection();    
21.	        }    
22.	  
23.	      public void getConnection(){    
24.	            System.out.println("获得连接...");    
25.	        }    
26.	            
27.	        public void releaseConnection(){    
28.	            System.out.println("释放连接...");    
29.	        }    
30.	}  

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

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

怎么样,现在理解了吧?

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

先说上面这段代码,本来完全可以用模板模式来进行实现:
Java代码

1.	public abstract class B{  
2.	     public void execute(){   
3.	            getConnection();    
4.	            doCRUD();    
5.	            releaseConnection();    
6.	        }    
7.	  
8.	      public abstract void doCRUD();  
9.	  
10.	      public void getConnection(){    
11.	            System.out.println("获得连接...");    
12.	        }    
13.	            
14.	        public void releaseConnection(){    
15.	            System.out.println("释放连接...");    
16.	        }    
17.	}  
18.	  
19.	public class A extends B{  
20.	    public void doCRUD(){    
21.	          System.out.println("执行add操作...");    
22.	     }    
23.	  
24.	     public void add(){    
25.	             doCRUD();  
26.	        }    
27.	}  
28.	  
29.	public class C extends B{  
30.	    public void doCRUD(){    
31.	          System.out.println("执行delete操作...");    
32.	     }    
33.	  
34.	     public void delete(){    
35.	             doCRUD();  
36.	        }    
37.	}  

如果改为回调实现是这样的:
Java代码

1.	interface CallBack{     
2.	    public void doCRUD();     
3.	}    
4.	    
5.	public class HibernateTemplate {     
6.	    public void execute(CallBack action){    
7.	        getConnection();    
8.	        action.doCRUD();    
9.	        releaseConnection();    
10.	    }    
11.	     
12.	    public void add(){    
13.	         execute(new CallBack(){    
14.	            public void doCRUD(){    
15.	                System.out.println("执行add操作...");    
16.	            }    
17.	         });    
18.	     }     
19.	  
20.	     public void delete(){    
21.	         execute(new CallBack(){    
22.	            public void doCRUD(){    
23.	                System.out.println("执行delete操作...");    
24.	            }    
25.	         });    
26.	     }   
27.	    
28.	    public void getConnection(){    
29.	        System.out.println("获得连接...");    
30.	    }    
31.	        
32.	    public void releaseConnection(){    
33.	        System.out.println("释放连接...");    
34.	    }    
35.	        
36.	}    

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

下面再给出一个关于利用回调配合异步调用的很不错的例子
回调接口:
Java代码

1.	public interface CallBack {    
2.	    /**  
3.	     * 执行回调方法  
4.	     * @param objects   将处理后的结果作为参数返回给回调方法  
5.	     */    
6.	    public void execute(Object... objects );    
7.	}    

消息的发送者:
Java代码

1.	/** 
2.	 * 这个类相当于你自己 
3.	 */  
4.	public class Local implements CallBack,Runnable{    
5.	     
6.	    private Remote remote;    
7.	        
8.	    /**  
9.	     * 发送出去的消息  
10.	     */    
11.	    private String message;    
12.	        
13.	    public Local(Remote remote, String message) {    
14.	        super();    
15.	        this.remote = remote;    
16.	        this.message = message;    
17.	    }    
18.	    
19.	    /**  
20.	     * 发送消息  
21.	     */    
22.	    public void sendMessage()    
23.	    {    
24.	        /**当前线程的名称**/    
25.	        System.out.println(Thread.currentThread().getName());    
26.	        /**创建一个新的线程发送消息**/    
27.	        Thread thread = new Thread(this);    
28.	        thread.start();    
29.	        /**当前线程继续执行**/    
30.	        System.out.println("Message has been sent by Local~!");    
31.	    }    
32.	    
33.	    /**  
34.	     * 发送消息后的回调函数  
35.	     */    
36.	    public void execute(Object... objects ) {    
37.	        /**打印返回的消息**/    
38.	        System.out.println(objects[0]);    
39.	        /**打印发送消息的线程名称**/    
40.	        System.out.println(Thread.currentThread().getName());    
41.	        /**中断发送消息的线程**/    
42.	        Thread.interrupted();    
43.	    }    
44.	        
45.	    public static void main(String[] args)    
46.	    {    
47.	        Local local = new Local(new Remote(),"Hello");    
48.	            
49.	        local.sendMessage();    
50.	    }    
51.	    
52.	    public void run() {    
53.	        remote.executeMessage(message, this);  //这相当于给同学打电话,打完电话之后,这个线程就可以去做其他事情了,只不过等到你的同学打回电话给你的时候你要做出响应  
54.	            
55.	    }    
56.	}    

消息的接收者:
Java代码

1.	/** 
2.	 * 这个类相当于你的同学 
3.	 */  
4.	public class Remote {    
5.	    
6.	    /**  
7.	     * 处理消息  
8.	     * @param msg   接收的消息  
9.	     * @param callBack  回调函数处理类  
10.	     */    
11.	    public void executeMessage(String msg,CallBack callBack)    
12.	    {    
13.	        /**模拟远程类正在处理其他事情,可能需要花费许多时间**/    
14.	        for(int i=0;i<1000000000;i++)    
15.	        {    
16.	                
17.	        }    
18.	        /**处理完其他事情,现在来处理消息**/    
19.	        System.out.println(msg);    
20.	        System.out.println("I hava executed the message by Local");    
21.	        /**执行回调**/    
22.	        callBack.execute(new String[]{"Nice to meet you~!"});  //这相当于同学执行完之后打电话给你  
23.	    }    
24.	        
25.	}    

由上面这个例子可见,回调可以作为异步调用的基础来实现异步调用。
29 

这是典型的策略模式,又带点模版模式的味道

Java代码
//模拟Spring中HibernateTemplate回调机制的代码

Java代码

1.	interface CallBack{          
2.	       public void doCRUD();          
3.	   }         
4.	            
5.	   public class HibernateTemplate {          
6.	                
7.	       public void execute(CallBack action){         
8.	           getConnection();         
9.	           action.doCRUD();         
10.	           releaseConnection();         
11.	       }         
12.	             
13.	       public void add(){         
14.	            execute(new CallBack(){         
15.	               public void doCRUD(){         
16.	                   System.out.println("执行add操作...");         
17.	               }         
18.	            });         
19.	            System.out.println("continue exec");    
20.	       }          
21.	            
22.	       public void getConnection(){         
23.	           System.out.println("获得连接...");         
24.	       }         
25.	                
26.	       public void releaseConnection(){         
27.	           System.out.println("释放连接...");         
28.	       }         
29.	                
30.	   }    

15 楼 kprrr 2012-05-24
仔细看了很久,想请教一个问题,这种回调机制,跟策略模式有什么区别吗?
14 楼 jkxydp 2011-07-22
其实际上并没有放下电话的过程,因为在同一线程中,执行b调a的方法,a再调b的方法,资源从不曾释放,等到b调用完成返回并退出a的方法后,才会释放!
13 楼 cantellow 2011-07-22
你最后举的那个例子很好,充分说明了面向对象建模的现实性,需求来源与现实,所以语言的建模也更加面向对象,面向现实。
12 楼 cantellow 2011-07-22
我觉得这样做是更加面向对象,只要深刻理解了面向对象的含义,可能你平时经常用到了这种机制,只是没有这么明确罢了。

A调用B的方法是一种代理,但是这个过程中,有些行为是必须依赖A的方法,所以在B的方法里,执行完它自己的一些操作之后,它会调用A的方法。

整个过程就是一个面向对象的建模,A的方法该执行什么操作,什么时候交给B来执行,B什么时候调用A,关键操作时放在A好还是放在B好,这都是面向对象建模,理解了这点理解回调就很容易了。

LZ的文章还是不错的。
11 楼 semmy 2011-07-22
Java代码

1.	//模拟Spring中HibernateTemplate回调机制的代码     
2.	    interface CallBack{        
3.	        public void doCRUD();        
4.	    }       
5.	           
6.	    public class HibernateTemplate {        
7.	               
8.	        public void execute(CallBack action){       
9.	            getConnection();       
10.	            action.doCRUD();       
11.	            releaseConnection();       
12.	        }       
13.	            
14.	        public void add(){       
15.	             execute(new CallBack(){       
16.	                public void doCRUD(){       
17.	                    System.out.println("执行add操作...");       
18.	                }       
19.	             });       
20.	             System.out.println("continue exec");  
21.	        }        
22.	           
23.	        public void getConnection(){       
24.	            System.out.println("获得连接...");       
25.	        }       
26.	               
27.	        public void releaseConnection(){       
28.	            System.out.println("释放连接...");       
29.	        }       
30.	               
31.	    }  

System.out.println(“continue exec”);
增加一句这个代码,可以看出,还是一直拿着电话在那里等。
10 楼 semmy 2011-07-22
引用
你有一个复杂的问题解决不了,打电话给你的同学,你的同学说可以解决这个问题,但是需要一些时间,那么你不可能一直拿着电话在那里等,你会把你的电话号码告诉他,让他解决之后打电话通知你。回调就是体现在你的同学又反过来拨打你的号码。

其实我觉得这就是一个异步操作的案例,但你所列的那些代码例子并没有有异步操作的。这样回调还是同步的,必须要拿着电话等。
9 楼 hlylove 2011-07-22
但是需要一些时间,那么你不可能一直拿着电话在那里等,
ak478288 写道
感觉lz对回调理解有误,他并不是为了解决耗时操作的问题,他解决的就是一个设计模式的问题,任何模板模式的应用都可以使用这种方式来实现,而且不只是模板模式,命令模式也可以。

lz所说的解决耗时操作应该是多线程的问题,异步调用。但是你所给的例子却是不是这个意思

楼主的打电话例子中
引用
但是需要一些时间,那么你不可能一直拿着电话在那里等,
就容易让人产生是异步调用
8 楼 smiky 2011-07-22
function(fn){
fn.call()
}
JAVA中这种使用太多了吧,一个方法接收一个接口参数,当调用这个方法时,就会执行参数的方法
典型使用就是线程吧,当调用start时就会执行线程的run方法

跟正常的操作没什么区别,还是javascript中说是回调还算正常,java中这个概念简直有强加的嫌疑
method(obj){
obj.do();
}
7 楼 ak478288 2011-07-22
感觉lz对回调理解有误,他并不是为了解决耗时操作的问题,他解决的就是一个设计模式的问题,任何模板模式的应用都可以使用这种方式来实现,而且不只是模板模式,命令模式也可以。

lz所说的解决耗时操作应该是多线程的问题,异步调用。但是你所给的例子却是不是这个意思
6 楼 hlylove 2011-07-22
引用
事实上,很多需要进行回调的操作是比较费时的,被调用者进行费时操作,然后操作完之后将结果回调给调用者。

我不认为回调操作跟费时有什么联系,使用了Spring提供了JDBC回调,使用它是因为连接数据库费时?我不这么认为。
5 楼 luckyostar 2011-07-22

学习了,挺不错的。赞一个
4 楼 whumartine 2011-07-22
学习了,不错。
3 楼 energykey 2011-07-22
支持一下。
建议列出更多适用的场景,或者结合spring的源码给出适用场景。
当然,你的例子足以支撑本文的主题。
if…better…
2 楼 huqiji 2011-07-22
不错,确实是好东西,分享万岁
1 楼 Jclick 2011-07-22
很不错,赞一个!

本文转自论坛,具体位置不详!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

赵健zj

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值