Java(Android)回调函数详解

一、前言

本周有位入行开发不久的朋友问我回调究竟是个什么概念,在网上看了很多的回调函数解释,但是越看越乱。虽然回调函数这个梗已经不新鲜了,这里还是用书面的形式记录下。

如果有了解的,就无需再看。

二、概念

概念上,这里引用百度百科的解释,如下:

回调函数就是一个通过函数指针调用的函数。如果你把函数的指针(地址)作为参数传递给另一个函数,当这个指针被用来调用其所指向的函数时,我们就说这是回调函数。回调函数不是由该函数的实现方直接调用,而是在特定的事件或条件发生时由另外的一方调用的,用于对该事件或条件进行响应。

百度百科的定义就是上面这样的,它提到了函数指针这个概念,函数指针一般是C/C++的专有名词。Java中也是有这个概念的,只是我们把它叫做引用,淡化了函数指针的概念,使用也更加简单。

这个概念的定义,看上去是有点绕,并且不容易理解。网上有很多的对回调函数的说明,比如,A类调用B类的方法B1,B类又调用A类的方法A1之类。这样的说法,其实看的也很迷茫。

因此,会在下文中对回调函数做一个层次分明的解释。

三、元素

根据概念,我们知道一个回调函数的调用流程是需要以下三个元素,并分别给它们一个名字:

1、回调函数本身——回调函数;

2、回调函数作为参数传入的函数——中间函数;

3、调用者(调用函数)——调用函数。

根据以上的命名,回调函数的流程就是:

(1)将回调函数的引用,传入到中间函数(这个也可以称之为回调函数的注册/订阅)。

(2)调用函数调用中间函数,触发回调函数的事件。

纯文字说明可能没有感觉,这里我们引入一个很简单的例子,来描述这一流程:

首先,你需要一个回调函数:

package com.callback;

public class Callback {
	
	public void call(){
		System.out.println("我是一个回调函数,当我被打印出来的时候,说明回调函数被触发了");
	}
}

其次,再来一个中间函数:

package com.callback;

public class Middle {

	//参数是回调函数所在类的引用
	public void mid(Callback callback){
		//触发回调函数
		callback.call();
	}
}

最后,是调用函数:

package com.callback;

public class Main {

	/**
	 * @param args
	 */
	public static void main(String[] args) {
         Middle m=new Middle();
         //回调函数注册,将Callback的引用传入中间函数
         m.mid(new Callback());
         
	}

}

运行一下,你就可以看到打印结果:

我是一个回调函数,当我被打印出来的时候,说明回调函数被触发了

以上是一个简单例子,在该例子中,用最简方式模仿了回调函数的调用过程。但日常中我们是不会或者很少这么用的,因为该例子没有很好的表现出回调函数的作用。

回调函数有什么作用?在百度百科上有一段对它们意义的说明:

1、因为可以把调用者与被调用者分开,所以调用者不关心谁是被调用者。它只需知道存在一个具有特定原型和限制条件的被调用函数

2、回调可用于通知机制。

3、这一设计允许了底层代码调用在高层定义的子程序。

简单的说,可以用于解耦、通知以及其他。

为了完整说明回调函数的作用,我们再原来的例子上,做个扩展,引入接口。扩展的还是上面的例子,引入接口:

新增接口:

package com.callback;

public interface CallbackInterface {

	public void call();
}

原有的回调函数,实现上面的接口:

package com.callback;

public class Callback implements CallbackInterface{

	public void call() {
		// TODO Auto-generated method stub
		System.out.println("我是一个回调函数,当我被打印出来的时候,说明回调函数被触发了");

	}
	
}

中间函数的参数,做以下修改(修改为接口):

package com.callback;

public class Middle {

	//参数是回调函数所在类的引用
	public void mid(CallbackInterface callback){
		//触发回调函数
		callback.call();
	}
}

最后是调用函数:

package com.callback;

public class Main {
	/**
	 * @param args
	 */
	public static void main(String[] args) {
         Middle m=new Middle();
         //回调函数注册
         CallbackInterface ifsImpl=new Callback();
         m.mid(ifsImpl);
	}

}
对比两个例子,在后面这个例子中,引入接口,实现了:调用函数、中间函数与回调函数的解耦。底层函数对高层函数的调用。通知机制。


四、回调函数作为参数传入的方法

这是一个小细节,设计人员可以仿照依赖注入的三种方式,来实现回调函数的参数传入。

1、构造函数中传入;

2、用set方式传入;

3、直接将函数(接口)作为参数传入;

例子中就是第三种方式。

到此,回调函数的介绍就基本结束了。但是,上面的两个例子和我们实际接触的还是有那么一些差距。

在看完上面的例子后,对回调有了比较深的认识。现在,已经可以理论联系实际了。

也举一个日常用的很多的场景:Android中的按钮点击事件(这也是应用最广泛的一个回调函数)。


五、Andriod按钮点击事件的模拟对比

1、新开一个Android工程,我们做一个真实的按钮点击事件,如下:

//原始
        Button btn=(Button)findViewById(R.id.btn);
        btn.setOnClickListener(new OnClickListener() {
			
			@Override
			public void onClick(View arg0) {
				// TODO Auto-generated method stub
				System.out.println("按钮被点击了-原始");
			}
		});

2、再根据前面我们自己的分析,做一个按钮点击的回调函数事件:

也首先是一个回调函数:

package com.example.callbackandroid;

public interface MyOnClickListener {
//回调函数
	public void onClick();
}

中间函数:

package com.example.callbackandroid;

public class MyButton {

	private MyOnClickListener listener;
	//回调函数注册/订阅/登记
	public void setOnclickListener(MyOnClickListener listener){
		this.listener=listener;
	}
	
	public void doOnclick(){
		listener.onClick();
	}
}

调用函数:

 //模拟
        MyButton mButton=new MyButton();
        mButton.setOnclickListener(new MyOnClickListener() {
			
			@Override
			public void onClick() {
				// TODO Auto-generated method stub
				System.out.println("按钮被点击了-原始");
			}
		});
        
        mButton.doOnclick();

对比原始的android点击事件以及我们模拟的android点击事件,发现有一点区别:原始的android点击不需要没有调用doOnclick事件,没有调用函数??

实际上,原始的android点击事件是有调用函数的,只是不需要写在这里而已。如果有看过Android的onClick事件(事件分发)源码的朋友,会发现:在源码中,当android系统检测到该View的ACTION_UP的操作时,会调用performClick()这个函数,而这个函数的内容如下:

public boolean performClick() {
    sendAccessibilityEvent(AccessibilityEvent.TYPE_VIEW_CLICKED);
    if (mOnClickListener != null) {
        playSoundEffect(SoundEffectConstants.CLICK);
        mOnClickListener.onClick(this);
        return true;
    }
    return false;
}
它的调用函数,是在这里实现的。只要mOnClickListener不为null(这个就是通过我们setOnClickListener来赋值的),那么onClick就会被调用。

因此,它和模拟点击原理其实是一样的。

到这里,回调函数的分析就告一段落了。

下一段落,主要是一个概念的整理。

六、回调函数与同步、异步

在真正使用回调函数的时,我们经常会接触到异步回调、异步调用、同步调用这些词。这里简单说下概念,以免混乱,作为结尾。

同步调用:
这个是日常使用最频繁的一种调用方法,这是一个阻塞调用,如你调用一个方法,等待这个方法返回,或者执行完毕。

异步调用:

这个在android中也使用频繁,非阻塞调用,如你调用一个方法,无需等这个方法返回,即往下执行其他操作。

异步回调:

这个是异步+回调的形式。





本页内容版权归属为原作者,如有侵犯您的权益,请通知我们删除。

Android_聊天_表情 - 2016-07-23 17:07:26

接下来就进入聊天界面了,我的界面效果如下几个图所示: 这其中包括两个点:仿微信按住说话功能,表情管理 第一个,按住说话 按钮的功能,通过重写Button完成, /** * 控制录音Button * 1、重写onTouchEvent;(changeState方法、wantToCancel方法、reset方法); * 2、编写AudioDialogManage、并与该类AudioRecorderButton进行整合; * 3、编写AudioManage、并与该类AudioRecorderButton进行整合;

JAVA 面向对象 隐藏和封装 - 2016-07-23 17:07:20

本页面更新日期: 2016年07月22日 前言 在前面程序中,经常出现通过 某个对象直接访问其成员变量的情况. 这可能引起一些潜在问题,比如将某个 Person 的 age 成员变量直接设为 1000. 这在语法上没有任何问题, 但显然违背了当前的自然规律. 人怎么可能活到 1000岁 - - . (就现在的科学来讲) Java也考虑到了这种情况, 为你提供了 类和对象的成员变量 进行封装的方法,来保护成员变量不被恶意修改. 理解封装 封装(Encapsulation) 是面向对象的三大特征之一.(另外两
上一篇文章分析过DroidPlugin对Activity的处理过程,不得不为对DroidPlugin的工程师们钦佩不已,那么是不是Service可以像Activity的处理过程一样来处理呢?前面讲过每一个代理进程只是预定义了一个Service,如果某一个插件中有多个Service,那岂不是某一个时刻只能有一个Service运行呢?由此可以判定可能Service的处理和Activity不一样。 一方面:平时使用Activity主要是用于展示界面和用户交互,Activity的生命周期可能受用户控制,当用户操作

IntentService使用及源码分析 - 2016-07-23 14:07:58

IntentService使用及源码分析 转载请注明 原博客地址: http://blog.csdn.net/gdutxiaoxu/article/details/52000680 本篇博客主要简介一下三个问题: 什么是IntentService? 怎样使用IntentService IntentSerice()源码分析 1)什么是IntentService? 我们知道Service和Activity一样是Android的四大组件之一,Service简称为后台服务,具有较高的优先级别。我们平时在Activ

Android——ListView控件 - 2016-07-23 14:07:54

本篇介绍ListView控件,这是Android中比较重要也比较复杂的控件,这里只谈到使用ViewHolder机制优化即可。 一、ListView简介 ListView是Android系统中显示列表的控件,每个ListView都可以包含很多个列表项。 二、ListView的使用 概念不多说,直接来介绍使用方法。 ListView中比较复杂的是数据适配器,其作用是把复杂的数据(数组、链表、数据库、集合等)填充在指定视图界面,是连接数据源和视图界面的桥梁。常见的Android原生的适配器有ArrayAdapt
HTTP请求报文: 一个HTTP请求报文由四个部分组成:请求行、请求头部、空行、请求数据 1.请求行   请求行由请求 方法字段、URL字段和HTTP协议版本字段 3个字段组成,它们用空格分隔。比如 GET /data/info.html HTTP/1.1 方法字段就是HTTP使用的请求方法,比如常见的GET/POST 其中HTTP协议版本有两种:HTTP1.0/HTTP1.1 可以这样区别: HTTP1.0对于每个连接都的建立一次连接一次只能传送一个请求和响应,请求就会关闭,HTTP1.0没有Host字

gradle多渠道打包 - 2016-07-23 14:07:48

E文不好的童鞋,例如我,翻译文章的过程里没有愉悦的感受,只有2行老泪;但最终有一丝成就感也算是安慰了。所以我会去尊重那些翻译IT技术文章的大拿们,他们就是千千万万个亚里士多德和吴启明,他们是E文不好的童鞋的传教士,阿门,当然我不是大拿。 废话少说,先看一篇例子:在 http://ghui.me/post/2015/03/create-several-variants/  。 然后来看这篇翻译,扫清例子中一部分未知的知识。原文在 http://tools.android.com/tech-docs/new-b
前言 本文主要讲解Telephony中Phone相关的知识,主要想讲明白三件事情: Phone是什么? Phone从哪里来? Phone有什么作用? 1. Phone是什么 1.1 Phone是一个接口 Phone.java (frameworks\opt\telephony\src\java\com\android\internal\telephony) public interface Phone { //包含了大量的register/unregister的方法。(监听能力) void registe

Android混淆心得 - 2016-07-23 14:07:36

最近在做Android应用的混淆,踩了一些坑,这里记录分享下个人的心得。 混淆介绍 首先先简单说一下什么是混淆和混淆的作用,其实这个搜索下可以找到一堆官方的说法等等,这里简单口语叙述一下,混淆就是把代码替换成a、b、c基本字母组成的代码,比如一个方法名为:function(),混淆后可能会被替换成a()。 混淆的好处: 代码混淆后阅读性降低,反编译后破译程序难度提高 混淆后字节数减少,减少了应用了体积 前者只能说有一点作用,后者则需要看代码的数量 当然不能忽视混淆的缺点: 混淆后,测试不充分可能导致某些功
[编写高质量iOS代码的52个有效方法](二)对象 参考书籍:《Effective Objective-C 2.0》 【英】 Matt Galloway 先睹为快 6.理解“属性”这一概念 7.在对象内部尽量直接访问实例变量 8.理解“对象等同性”这一概念 9.以“类簇模式”隐藏实现细节 10.在既有类中使用关联对象存放自定义数据 目录 编写高质量iOS代码的52个有效方法二对象 先睹为快 目录 第6条理解属性这一概念 第7条在对象内部尽量直接访问实例变量 第8条理解对象等同性这一概念 第9条以类簇模式隐