前言

回调的核心就是回调方将本身即this传递给调用方,这样调用方就可以在调用完毕之后告诉回调方它想要知道的信息。

1、什么是回调

软件模块之间总是存在一定的接口,从调用方式上,可以把他们分为三类:同步调用、回调和异步调用。

(1)同步调用:

同步调用是最基本并且最简单的一种调用方式,类A的方法a()调用类B的方法b(),一直等待b()方法执行完毕,a()方法再继续往下走。这种调用方式适用于方法b()执行时间不长的情况,因为b()方法执行时间一长或者直接阻塞的话,a()方法的余下代码是无法执行下去的,这样会造成整个流程的阻塞。

(2)异步调用:

是一种类似消息或事件的机制,是为了解决同步调用可能出现阻塞,导致整个流程卡住而产生的一种调用方式。类A的方法a()通过新起线程的方式调用类B的方法b(),代码接着直接往下执行,这样无论方法b()执行时间多久,都不会阻塞方法a()的执行。但是这种方式,由于方法a()不等待方法b()执行完成,在方法a()需要方法b()执行结果的情况下(视具体业务而定,有些业务比如启动异步线程发个微信通知、刷新一个缓存这种就没有必要),必须通过一定的方法对方法b()的执行结果进行监听。在Java中,可以使用Future Callable的方式做到这一点。

(3)回调:

最后是回调,回调的思想是:

类A的a()方法调用了类B的b()方法类B的b方法执行完毕主动调用类A的callback()方法

这样一种调用方式组成了上图,也就是一种双向的调用方式

回调函数是一个函数或过程,不过它是一个由调用方自己实现,供被调用方使用的特殊函数。

在面向对象的语言中,回调则是通过接口或抽象类来实现的,我们把实现这种接口的类称为回调类,回调类的对象称为回调对象。

2、例子

开始之前。先想象一个场景:幼稚园的小朋友刚刚学习了10以内的加法。

第一章.故事的缘起

幼师在黑板上写一个式子 “1 1=”,由小明来填空由于已经学习了10以内的加法,小明同学可以完全靠自己来计算这个题目,模拟该过程的代码如下:

public class Student
{
	private String name=null;
	public Student(String name)
	{
		this.name=name;
	}
	public void setName(String name)
	{
		this.name=name;
	}
	private int calcADD(int a,int b)
	{
		return a b;
	}
	public void fillBlank(int a,int b)
	{
		int result=calcADD(a,b);
		System.out.println(name "心算:" a " " b "=" result);
	}
}

小明同学在填空(fillBlank)的时候,直接心算(clacADD)了一下,得出结果是2,并将结果写在空格里。

测试代码如下:

public class Test
{
	public static void main(String[] args)
	{
		int a=1;
		int b=1;
		Student s=new Student("小明");
		s.fillBlank(a,b);
	}
}

运行结果如下:

小明心算:1 1=2

该过程完全由Student类的实例对象单独完成,并未涉及回调机制。

第二章.幼师的找茬

课间,幼师突发奇想在黑板上写了"168 291=",让小明完成,然后回办公室去了。
这时候小明明显不能再像上面那样靠心算来完成了,正在懵逼的时候,班上的小红同学递过来一个只能计算加法的计算机,而小明同学恰好知道怎么用计算器,于是通过计算器计算得到结果并完成了填空。

计算器的代码为:

public class Calculator
{
	public int add(int a,int b)
	{
		return a b;
	}
}

修改Student类,添加使用计算器的方法:

public class Student
{
	private String name=null;
	public Student(String name)
	{
		this.name=name;
	}
	public void setName(String name)
	{
		this.name=name;
	}
	private int calcADD(int a,int b)
	{
		return a b;
	}
	private int useCalculator(int a,int b)
	{
		return new Calculator().add(a,b);
	}
	public void fillBlank(int a,int b)
	{
		int result=useCalculator(a,b);
		System.out.println(name "使用计算器:" a " " b "=" result);
	}
}

测试代码如下:

public class Test
{
	public static void main(String[] args)
	{
		int a=168;
		int b=291;
		Student s=new Student("小明");
		s.fillBlank(a,b);
	}
}

运行结果如下:

小明使用计算器:168 291=459

该过程中仍未涉及到回调机制,但是小明的部分工作已经实现了转移,由计算器来协助实现。

第三章.幼师回来了

发现小明完成了3位数的加法,老师觉得小明很聪明,是个可塑之才。于是又在黑板上写下了"26549 16387=",让小明上课之前完成填空,然后又回办公室了。
小明看着小红再一次递上来的计算机,心生一计:让小红代劳。
小明告诉小红题目是"26549 16487=",然后指出填写结果的具体位置,然后就出去快乐的玩耍了。
这里,不把小红单独实现出来,而是把这个只能算加法的计算器和小红看成一个整体,一个会算结果还会填空的超级计算器。折这个超级计算器需要传的参数是两个加数和要填空的位置,而这些内容需要小明提前告知,也就是小明要把自己的一部分方法暴露给小红,最简单的方法就是把自己的引用和两个加数一块告诉小红。因此,超级计算器的add方法应该包含两个操作数和小明自身的引用,

代码如下:

public class SuperCalculator
{
	public void add(int a,int b,Student xiaoming)
	{
		int result=a b;
		xiaoming.fillBlank(a,b,result);
	}
}

小明这边现在已经不需要心算,也不需要使用计算器,因此只需要有一个方法可以向小红寻求帮助就行了,

代码如下:

public class Student
{
	private String name=null;
	public Student(String name)
	{
		this.name=name;
	}
	public void setName(String name)
	{
		this.name=name;
	}
	public void callHelp(int a,int b)
	{
		new SuperCalculator().add(a,b,this);
	}
	public void fillBlank(int a,int b,int result)
	{
		System.out.println(name "求助小红计算:" a " " b "=" result);
	}
}

测试代码如下:

public class Test
{
	public static void main(String[] args)
	{
		int a=26549;
		int b=16487;
		Student s=new Student("小明");
		s.callHelp(a,b);
	}
}

运行结果为:

小明求助小红计算:26549 16487=43036

执行流程为:小明通过自身的callHelp方法调用了小红(new SuperCalculator())的add方法,在调用的时候将自身的引用(this)当作参数一并传入,小红在使用计算器得出结果之后,回调了小明的fillBlank方法,将结果填在了黑板的空格上。

到这里,回调功能就正式登场了,小明的fillBlank方法就是我们常说的回调函数。

通过这种方式,可以明显的看出,对于完成老师的填空题这个问题上,小明已经不需要等待到加法做完且结果填写在黑板上才能去跟小伙伴撒欢了,填空这个工作由超级计算器小红来做了。回调的优势已经开始体现了。

第四章.门口的婆婆

幼稚园的门口有一个头发花白的老婆婆,每天风雨无阻在那里摆着地摊卖一些快过期的垃圾食品。由于年纪大了,脑子有些糊涂,经常算不清楚自己挣了多少钱。有一天,她无意间听到了小明跟小伙伴们吹嘘自己如何在小红的帮助下与幼师斗智斗勇。于是,婆婆决定找到小红牌超级计算器来做自己的小帮手,并提供一包卫龙辣条作为报酬。小红经不住诱惑,答应了。

回看一下上一章的代码,我们发现小红牌超级计算器的add方法需要的参数是两个整型变量和一个Student对象,但是老婆婆她不是学生,是个小商贩啊,这里肯定要做修改。这种情况下,我们很自然的会想到继承和多态。如果让小明这个学生和老婆婆这个小商贩从一个父类进行继承,那么我们只需要给小红牌超级计算器传入一个父类的引用就可以啦。

不过,实际使用中,考虑到java的单继承,以及不希望把自身太多东西暴漏给别人,这里使用从接口继承的方式配合内部类来做。

换句话说,小红希望以后继续向班里的小朋友们提供计算服务,同时还能向老婆婆提供算账服务,甚至以后能够拓展其他人的业务,于是她向所有的顾客约定了一个办法,用于统一的处理,也就是自己需要的操作数和做完计算之后应该怎么做。这个统一的方法,小红做成了一个接口,提供给了大家,

代码如下:

public interface doJob
{
	public void fillBlank(int a,int b,int result);
}

因为灵感来自帮小明填空,因此小红保留了初心,把所有业务当做填空(fillBlank)来做。

同时,小红修改了自己的计算器,使其可以同时处理不同的实现了doJob接口的人,代码如下:

public class SuperCalulator
{
	public void add(int a,int b doJob customer)
	{
		int result=a b;
		customer.fillBlank(a,b,result);
	}
}

小明和老婆婆拿到这个接口之后,只要实现了这个接口,就相当于按照统一的模式告诉小红得到结果之后的处理办法,按照之前说的使用内部类来做,代码如下:

小明的:

public class Student
{
	private String name=null;
	public Student(String name)
	{
		this.name=name;
	}
	public class doHomeWork implements doJob
	{
		@Override
		public void fillBlank(int a,int b,int result)
		{
			System.out.println(name "求助小红计算:" a " " b "=" result);
		}
	}
	public void callHelp(int a,int b)
	{
		new SuperCalculator().add(a,b,new doHomeWork());
	}
}

老婆婆的:

public class Seller
{
	private String name=null;
	public Seller(String name)
	{
		this.name=name;
	}
	public setName(String name)
	{
		this.name=name;
	}
	public class doHomeWork implements doJob
	{
		@Override
		public void fillBlank(int a,int b,int result)
		{
			System.out.println(name "求助小红算账:" a " " b "=" result "元";)
		}
	}
	public void callHelp(int a,int b)
	{
		new SuperCalculator().add(a,b,new doHomeWork());
	}
}

测试程序如下:

public class Test
{
	public static void main(String[] args)
	{
		int a=56;
		int b=31;
		int c=26497;
		int d=11256;
		Student s1=new Student("小明");
		Seller s2=new Seller("老婆婆");

		s1.callHelp(a,b);
		s2.callHelp(c,d);
	}
}

运行结果如下:

小明求助小红计算:56 31=87
老婆婆求助小红算账:26497 11256=37753元

到此这篇关于Java 回调callback举例详解的文章就介绍到这了,更多相关Java 回调内容请搜索Devmax以前的文章或继续浏览下面的相关文章希望大家以后多多支持Devmax!

Java 回调callback举例详解的更多相关文章

  1. html5录音功能实战示例

    这篇文章主要介绍了html5录音功能实战示例的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

  2. ios – CLGeocoder错误. GEOErrorDomain代码= -3

    有没有关于apple的地理编码请求的文档?谢谢你提前.更新这是我的整个代码请求解决方法在搜索到答案后,它在Apples文档中!

  3. ios – 调用异步方法的方法的单元测试

    我想我有这些代码行:我想为该代码编写一个单元测试.对于initializeHomeData和initializeAnythingElse,我可以编写单元测试,如:我的问题是,如何测试reset()?我应该在testReset()中调用它们,如:但我认为这不是适当的实施.解决方法你是对的.要测试重置,您需要调用reset,而不是内部方法.话虽这么说,重置目前的编写方式使其不可测试.您能够如此轻松地测

  4. swift - The Facade Pattern

    Facade(外观)模式为子系统中的各类提供一个简明一致的界面,隐藏子系统的复杂性,使子系统更加容易使用。

  5. swift - The Proxy Pattern

    我在实际工作中vc也仿照过Foundation的delegate:button:内涵业务逻辑,底层实现;每个button是一个类,业务逻辑需要未知的参数和处理之后未知的结果反馈UI:点击button之后界面的改变,UI实现未知的参数和未知的结果反馈,也就是实现这个代理这样以来UI的定制,很灵活很容易,代码思路依然清晰如初。哪个是主体哪个是代理并不重要关键是看定义所说whichisusedwhenanobjectisrequiredtoactasaninterfacetoanotherobjectorres

  6. swift 闭包的使用

    使用:定义:

  7. swift(ios) webview 的优化

    最近一直在做手机H5的东西,网页写多了,测试也测出问题来了,打开十几个网页后,app出现无响应,app的webview界面出现黑屏等等奇怪的问题。我试了几遍,APP内存占用从20M飙升到100M+,到了100M的时候,xcode被断开了,然后问题就一个个冒出来了-_-!搜索了一下,是uiwebview内存泄漏,然后我就兼容了wkwebview。

  8. swift闭包的使用 -- 类似于OC中的Block

    //点击按钮触发该方法funcbuttonClicked(){if(self.callBack!

  9. Swift2到Swift3语法变化不完整总结

    Swift3语法变化Swift3和Swift2对比,更加安全和更加的面向对象了Swift3废弃了旧版本的C类型的GCD写法,而换成了更加面向对象的全新的GCD写法Swift3闭包在方法内部使用限制的关键字修改为了@noescape(默认值)和@escaping如果在方法内部执行了另一个闭包需要使用到方法参数的闭包,需要加上@escaping关键字Swift3闭包的使用避免循环引用Swift3处理S

  10. 可点击 @、# 标记文本实现

    在社交类APP中@、#符号构成的标记文本已经形成了某种通用的意义:前者表示通知某位好友,而后者表示为某个话题或者分类。开始上码的代码首先声明了一个wordType的枚举类型,该类用用于对标示文本进行类型标记。这里之所以使用.character而不是后面的.word的原因是:后者会将@、#这些标示符丢弃,导致一只类似点击到无效区域的情形。当上诉检查通过也就是点击区域有效的时候,我们使用.word,获取点击区域的单词。

随机推荐

  1. 基于EJB技术的商务预订系统的开发

    用EJB结构开发的应用程序是可伸缩的、事务型的、多用户安全的。总的来说,EJB是一个组件事务监控的标准服务器端的组件模型。基于EJB技术的系统结构模型EJB结构是一个服务端组件结构,是一个层次性结构,其结构模型如图1所示。图2:商务预订系统的构架EntityBean是为了现实世界的对象建造的模型,这些对象通常是数据库的一些持久记录。

  2. Java利用POI实现导入导出Excel表格

    这篇文章主要为大家详细介绍了Java利用POI实现导入导出Excel表格,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

  3. Mybatis分页插件PageHelper手写实现示例

    这篇文章主要为大家介绍了Mybatis分页插件PageHelper手写实现示例,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪

  4. (jsp/html)网页上嵌入播放器(常用播放器代码整理)

    网页上嵌入播放器,只要在HTML上添加以上代码就OK了,下面整理了一些常用的播放器代码,总有一款适合你,感兴趣的朋友可以参考下哈,希望对你有所帮助

  5. Java 阻塞队列BlockingQueue详解

    本文详细介绍了BlockingQueue家庭中的所有成员,包括他们各自的功能以及常见使用场景,通过实例代码介绍了Java 阻塞队列BlockingQueue的相关知识,需要的朋友可以参考下

  6. Java异常Exception详细讲解

    异常就是不正常,比如当我们身体出现了异常我们会根据身体情况选择喝开水、吃药、看病、等 异常处理方法。 java异常处理机制是我们java语言使用异常处理机制为程序提供了错误处理的能力,程序出现的错误,程序可以安全的退出,以保证程序正常的运行等

  7. Java Bean 作用域及它的几种类型介绍

    这篇文章主要介绍了Java Bean作用域及它的几种类型介绍,Spring框架作为一个管理Bean的IoC容器,那么Bean自然是Spring中的重要资源了,那Bean的作用域又是什么,接下来我们一起进入文章详细学习吧

  8. 面试突击之跨域问题的解决方案详解

    跨域问题本质是浏览器的一种保护机制,它的初衷是为了保证用户的安全,防止恶意网站窃取数据。那怎么解决这个问题呢?接下来我们一起来看

  9. Mybatis-Plus接口BaseMapper与Services使用详解

    这篇文章主要为大家介绍了Mybatis-Plus接口BaseMapper与Services使用详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪

  10. mybatis-plus雪花算法增强idworker的实现

    今天聊聊在mybatis-plus中引入分布式ID生成框架idworker,进一步增强实现生成分布式唯一ID,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

返回
顶部