模板方法模式很常用,其目的是在一个方法中定义一个算法骨架,而将一些步骤延迟到子类中。模板方法使得子类可以在不改变算法结构的情况下,重新定义算法的某些步骤。在标准的模板方法模式实现中,主要是使用继承的方式,来让父类在运行期间可以调用到子类的方法。 详见抽象类部分示例。
其实在Java开发中,还有另外一个方法可以实现同样的功能,那就是Java回调技术。回调是一种双向的调用模式,也就是说,被调用的接口被调用时也会调用对方的接口,简单点说明就是:A类中调用B类中的C方法,然后B类中的C方法中反过来调用A类中的D方法,那么D这个方法就叫回调方法。
回调的具体过程如下:
1. Class A实现接口CallBack —— 背景1
2. class A中包含class B的引用 ——背景2
3. class B有一个参数为CallBack的方法C ——背景3
4. 前三条是我们的准备条件,接下来A的对象调用B的方法C
5. 然后class B就可以在C方法中调用A的方法D
这样说大家可能还是不太理解,下面我们根据示例5-33来说明回调机制。该示例的生活背景为:有一天小刘遇到一个很难的问题“学习Java选哪呢?”,于是就打电话问小高,小高一时也不太了解行情,就跟小刘说,我现在还有事,等忙完了给你咨询咨询,小刘也不会傻傻的拿着电话去等小高的答案,于是小刘对小高说,先挂电话吧,你知道答案后再打我电话告诉我吧,于是挂了电话。小高先去办自己的事情去了,过了几个小时,小高打电话给小刘,告诉他答案是“学Java当然去北京”。
【示例5-33】回调机制示例
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
|
/** * 回调接口 */ interface CallBack { /** * 小高知道答案后告诉小刘时需要调用的方法,即回调方法 * @param result 是问题的答案 */ public void answer(String result); } /** * 小刘类:实现了回调接口CallBack(背景一) */ class Liu implements CallBack { /** * 包含小高对象的引用 (背景二) */ private Gao gao; public Liu(Gao gao){ this .gao = gao; } /** * 小刘通过这个方法去问小高 * @param question 小刘问的问题“学习Java选哪呢?” */ public void askQuestion(String question){ //小刘问小高问题 gao.execute(Liu. this , question); } /** * 小高知道答案后调用此方法告诉小刘 */ @Override public void answer(String result) { System.out.println( "小高告诉小刘的答案是:" + result); } } /** * 小高类 */ class Gao { /** * 相当于class B有一个参数为CallBack的方法C(背景三) */ public void execute(CallBack callBack, String question){ System.out.println( "小刘问的问题是:" + question); //模拟小高挂点后先办自己的事情花了很长时间 try { Thread.sleep( 10000 ); } catch (InterruptedException e) { e.printStackTrace(); } //小高办完自己的事情后想到了答案 String result = "学Java当然去北京" ; //小高打电话把答案告诉小刘,相当于class B 反过来调用class A 的D方法 callBack.answer(result); } } public class Test { public static void main(String[] args) { Gao gao= new Gao(); Liu liu = new Liu(gao); //小刘问问题 liu.askQuestion( "学习Java选哪呢?" ); } } |
执行结果如图5-34所示:
图5-34 示例5-33运行效果图
通过回调在接口中定义的方法,调用到具体的实现类中的方法,其本质是利用Java的动态绑定技术,在这种实现中,可以不把实现类写成单独的类,而使用内部类或匿名内部类来实现回调方法。