这里,不把小红单独实现出来,而是把这个只能算加法的计算器和小红看成一个整体,一个会算结果还会填空的超级计算器。这个超级计算器需要传的参数是两个加数和要填空的位置,而这些内容需要小明提前告知,也就是小明要把自己的一部分方法暴漏给小红,最简单的方法就是把自己的引用和两个加数一块告诉小红。
因此,超级计算器的add方法应该包含两个操作数和小明自身的引用,代码如下:
SuperCalculator 2 { add(int a, int b, Student xiaoming) 4 { 5 int result = a + b; 6 xiaoming.fillBlank(a, b, result); 7 } 8 }
小明这边现在已经不需要心算,也不需要使用计算器了,因此只需要有一个方法可以向小红寻求帮助就行了,代码如下:
Student 2 { 3 private String name = null; Student(String name) 6 { .name = name; 9 } setName(String name) 12 { 13 this.name = name; 14 } callHelp (int a, int b) 17 { 18 new SuperCalculator().add(a, b, this); 19 } fillBlank(int a, int b, int result) 22 { 23 System.out.println(name + "求助小红计算:" + a + " + " + b + " = " + result); 24 } 25 }
测试代码如下:
Test 2 { main(String[] args) 4 { 5 int a = 26549; 6 int b = 16487; 7 Student s = new Student("小明"); 8 s.callHelp(a, b); 9 } 10 }
运行结果为:
小明求助小红计算:26549 + 16487 = 43036
执行流程为:小明通过自身的callHelp方法调用了小红(new SuperCalculator())的add方法,在调用的时候将自身的引用(this)当做参数一并传入,小红在使用计算器得出结果之后,回调了小明的fillBlank方法,将结果填在了黑板上的空格里。
灯灯灯!到这里,回调功能就正式登场了,小明的fillBlank方法就是我们常说的回调函数。
通过这种方式,可以很明显的看出,对于完成老师的填空题这个任务上,小明已经不需要等待到加法做完且结果填写在黑板上才能去跟小伙伴们撒欢了,填空这个工作由超级计算器小红来做了。回调的优势已经开始体现了。
第4章. 门口的婆婆
幼稚园的门口有一个头发花白的老婆婆,每天风雨无阻在那里摆着地摊卖一些快过期的垃圾食品。由于年纪大了,脑子有些糊涂,经常算不清楚自己挣了多少钱。有一天,她无意间听到了小明跟小伙伴们吹嘘自己如何在小红的帮助下与幼师斗智斗勇。于是,婆婆决定找到小红牌超级计算器来做自己的小帮手,并提供一包卫龙辣条作为报酬。小红经不住诱惑,答应了。
回看一下上一章的代码,我们发现小红牌超级计算器的add方法需要的参数是两个整型变量和一个Student对象,但是老婆婆她不是学生,是个小商贩啊,这里肯定要做修改。这种情况下,我们很自然的会想到继承和多态。如果让小明这个学生和老婆婆这个小商贩从一个父类进行继承,那么我们只需要给小红牌超级计算器传入一个父类的引用就可以啦。
不过,实际使用中,考虑到java的单继承,以及不希望把自身太多东西暴漏给别人,这里使用从接口继承的方式配合内部类来做。
换句话说,小红希望以后继续向班里的小朋友们提供计算服务,同时还能向老婆婆提供算账服务,甚至以后能够拓展其他人的业务,于是她向所有的顾客约定了一个办法,用于统一的处理,也就是自己需要的操作数和做完计算之后应该怎么做。这个统一的方法,小红做成了一个接口,提供给了大家,代码如下:
doJob 2 { fillBlank(int a, int b, int result); 4 }
因为灵感来自帮小明填空,因此小红保留了初心,把所有业务都当做填空(fillBlank)来做。
同时,小红修改了自己的计算器,使其可以同时处理不同的实现了doJob接口的人,代码如下:
SuperCalculator 2 { add(int a, int b, doJob customer) 4 { 5 int result = a + b; 6 customer.fillBlank(a, b, result); 7 } 8 }
小明和老婆婆拿到这个接口之后,只要实现了这个接口,就相当于按照统一的模式告诉小红得到结果之后的处理办法,按照之前说的使用内部类来做,代码如下:
小明的:
Student 2 { 3 private String name = null; Student(String name) 6 { .name = name; 9 } setName(String name) 12 { 13 this.name = name; 14 } doHomeWork implements doJob 17 { 18 19 @Override fillBlank(int a, int b, int result) 21 { System.out.println(name + "求助小红计算:" + a + " + " + b + " = " + result); 24 } 25 26 } callHelp (int a, int b) 29 { 30 new SuperCalculator().add(a, b, new doHomeWork()); 31 } 32 }
老婆婆的: