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中也使用频繁,非阻塞调用,如你调用一个方法,无需等这个方法返回,即往下执行其他操作。

异步回调:

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

时间: 2024-10-13 12:38:20

Java(Android)回调函数详解的相关文章

实例介绍 Java(android) 回调函数使用方法

在Android开发中经常用到回调机制,其中最典型的就是控件被触发的实现方式,简单而言,如Button被Click后,是系统调用了OnClick方法,而我们为Button注册了OnClickListener监听器,当被触发Click后,OnClickListener中的OnClick方法就会被回调,我们就能在其中执行相应操作了. 下面举一个简单的例子介绍回调的实现方式: 回调函数使用的简单例子 程序员A写了一段程序(程序a),其中预留有回调函数接口,并封装好了该程序.程序员B要让a调用自己的程序

Cocos2d-x3.1回调函数详解

Cocos2d-x3.1中回调函数的定义在CCRef.h中声明,源码如下: typedef void (Ref::*SEL_CallFunc)(); typedef void (Ref::*SEL_CallFuncN)(Node*); typedef void (Ref::*SEL_CallFuncND)(Node*, void*); typedef void (Ref::*SEL_CallFuncO)(Ref*); typedef void (Ref::*SEL_MenuHandler)(Re

Java回调函数详解

为了了解什么是回调函数,在网上查阅了如下资料,整理如下: 资料一: 首先说说什么叫回调函数? 在WINDOWS中,程序员想让系统DLL调用自己编写的一个方法,于是利用DLL当中回调函数(CALLBACK)的接口来编写程序,使它调用,这个就 称为回调.在调用接口时,需要严格的按照定义的参数和方法调用,并且需要处理函数的异步,否则会导致程序的崩溃. 这样的解释似乎还是比较难懂,这里举个简 单的例子: 程序员A写了一段程序(程序a),其中预留有回调函数接口,并封装好了该程序.程序员B要让a调用自己的程

JS中的回调函数详解

function studyEnglish(who){ document.write(who+"学习英语</br>"); } function study(callback,params){ callback(params); } study(studyEnglish,'ck'); //或者 function study(callback,params){ callback(params); } study(function(who){ document.write(who

jQuery 源码解析(八) 异步队列模块 Callbacks 回调函数详解

异步队列用于实现异步任务和回调函数的解耦,为ajax模块.队列模块.ready事件提供基础功能,包含三个部分:Query.Callbacks(flags).jQuery.Deferred(funct)和jQuery.when().本节讲解Callbacks,也就是回调函数列表 回调函数用于管理一组回调函数,支持添加.移除.触发.锁定和禁用回调函数,为jQuery.ajax.jQuery.Deferred()和ready()事件提供基础功能,我们也可以基于它编写新的组件. 使用方法:$.Callb

[应用相关] C 语言回调函数详解

1. 什么是回调函数? 回调函数,光听名字就比普通函数要高大上一些,那到底什么是回调函数呢?恕我读得书少,没有在那本书上看到关于回调函数的定义.我在百度上搜了一下,发现众说纷纭,有很大一部分都是使用类似这么一个场景来说明:A君去B君店里买东西,恰好缺货,A君留下号码给B君,有货时通知A君.感觉这个让人更容易想到的是异步操作,而不是回调.另外还有两句英文让我印象深刻:1) If you call me, I will call you back; 2) Don't call me, I will

回调函数详解

回调函就是一个函数调用另一个函数的过程,在编码过程中我们经常会遇到在上一个函数执行完后,才开始执行下一个函数. 下面是代码生成一个div然后让他的左边距增加到100然后停止. function fun1(){ var div =document.createElement('div'); div.style.width = '100px'; div.style.height = '100px'; div.style.background ='red'; document.body.appendC

Android 接口回调机制详解

在使用接口回调的时候发现了一个经常犯的错误,就是回调函数里面的实现有可能是用多线程或者是异步任务去做的,这就会导致我们期望函数回调完毕去返回一个主函数的结果,实际发现是行不通的,因为如果回调是多线程的话你是无法和主函数同步的,也就是返回的数据是错误的,这是非常隐秘的一个错误.那有什么好的方法去实现数据的线性传递呢?先介绍下回调机制原理. 回调函数 回调函数就是一个通过函数指针调用的函数.如果你把函数的指针(地址)作为参数传递给另一个函数,当这个指针被用为调用它所指向的函数时,我们就说这是回调函数

Android总结篇系列:Activity中几个主要函数详解

专注Android领域开发. 仰望星空,同时需要脚踏实地. ——好记性不如烂博客 Android总结篇系列:Activity中几个主要函数详解 Activity作为Android系统中四大基本组件之一,包含大量的与其他的各大组件.intent.widget以及系统各项服务等之间的交互的函数.在此,本文主要选取实际项目开发中常用的,但完全理解又需要有一定深入了解的几个函数进行讲解,后续本文会根据需要不断更新. 1. startActivityForResult / onActivityResult