Android回调机制浅谈

参考博客如下
http://blog.csdn.net/xiaanming/article/details/8703708
http://www.jianshu.com/p/3f86b7949f20
http://blog.csdn.net/xsf50717/article/details/50520462
这是看了这三篇关于回调的博客后所做的总结,新手小白,不好的地方见谅见谅!

经典回调方式

  • class A实现接口CallBack callback——背景1
  • class A中包含一个class B的引用b ——背景2
  • class B有一个参数为callback的方法f(CallBack callback) ——背景3
  • A的对象a调用B的方法 f(CallBack callback) ——A类调用B类的某个方法 C
  • 然后b就可以在f(CallBack callback)方法中调用A的方法 ——B类调用A类的某个方法D

下面举一个小例子

/*定义一个接口Callback*/
interface Callback{
    void doCallback(String result);
}
/*A类实现接口*/
class A implements Callback{
/*A类持有B类的引用,并把b传入A*/
    B b;
    A(B b){
        this.b = b;
    }
    /*定义A类方法*/
    public void funtionA(){
        //此时b方法传入的参数必须是Callback的子类,这里我们传入A,因为A实现了Callback
        b.setOnClickListener(A.this);
    }
    
    @Override
    public void doCallback(String result) {
        System.out.println("the answer is:"+result);
        
    }

}

class B {
    //创建一个含有Callback实例的B类方法
    public void setOnClickListener(Callback callback){
        String answer = "callback";
        /*B类方法中调用A类方法,这个A类方法要包括接口中的那个抽象方法,即doCallback(),并将所得结果作为参数传回A类中
         * 这里就是回调,实现内容在B类的回调中完成
         */
         callback.doCallback(answer);
    }
}

public class CallBackTest{
    public static void main(String [] args){
        B b = new B();
        A a = new A(b);
        //调用A类方法进回调
        a.funtionA();
    }
}

流程分析:
  • 主函数执行到a.funtionA();
  • 调用b.setOnClickListener(a);
  • String answer = "callback";表示在b中完成了该流程得到结果
  • a.doCallback(answer);然后将结果以参数的形式传入a中,执行doCallback()方法,回调
  • System.out.println("the answer is:"+result);//a.doCallback(answer);执行后进行结果输出
button中onclick()的第一种写法
//这个是View的一个回调接口  
/** 
 * Interface definition for a callback to be invoked when a view is clicked. 
 */  
public interface OnClickListener {  
    /** 
     * Called when a view has been clicked. 
     * 
     * @param v The view that was clicked. 
     */  
    void onClick(View v);  
}  
package com.example.demoactivity;  
  
import android.app.Activity;  
import android.os.Bundle;  
import android.view.View;  
import android.view.View.OnClickListener;  
import android.widget.Button;  
import android.widget.Toast;  
  
/** 
 * 这个就相当于Class A 
 * @author xiaanming 
 * 实现了 OnClickListener接口---->背景一 
 */  
public class MainActivity extends Activity implements OnClickListener{  
    /** 
     * Class A 包含Class B的引用----->背景二 
     */  
    private Button button;  
  
    @Override  
    public void onCreate(Bundle savedInstanceState) {  
        super.onCreate(savedInstanceState);  
        setContentView(R.layout.activity_main);  
        button = (Button)findViewById(R.id.button1);  
          
        /** 
         * Class A 调用View的方法,而Button extends View----->A类调用B类的某个方法 C 
         */  
        button.setOnClickListener(this);  
    }  
  
    /** 
     * 用户点击Button时调用的回调函数,你可以做你要做的事 
     * 这里我做的是用Toast提示OnClick 
     */  
    @Override  
    public void onClick(View v) {  
        Toast.makeText(getApplication(), "OnClick", Toast.LENGTH_LONG).show();  
    }  
  
}  
/** 
 * 这个View就相当于B类 
 * @author xiaanming 
 * 
 */  
public class View implements Drawable.Callback, KeyEvent.Callback, AccessibilityEventSource {  
    /** 
     * Listener used to dispatch click events. 
     * This field should be made private, so it is hidden from the SDK. 
     * {@hide} 
     */  
    protected OnClickListener mOnClickListener;  
      
    /** 
     * setOnClickListener()的参数是OnClickListener接口------>背景三 
     * Register a callback to be invoked when this view is clicked. If this view is not 
     * clickable, it becomes clickable. 
     * 
     * @param l The callback that will run 
     * 
     * @see #setClickable(boolean) 
     */  
      
    public void setOnClickListener(OnClickListener l) {  
        if (!isClickable()) {  
            setClickable(true);  
        }  
        mOnClickListener = l;  
    }  
      
      
    /** 
     * Click的触发是在系统捕捉到ACTION_UP后发生并由performClick()执行的,
     *performClick里会调用先前注册的监听器的onClick()方法:      
     */  
    public boolean performClick() {  
        sendAccessibilityEvent(AccessibilityEvent.TYPE_VIEW_CLICKED);  
  
        if (mOnClickListener != null) {  
            playSoundEffect(SoundEffectConstants.CLICK);  
              
            //这个不就是相当于B类调用A类的某个方法D,这个D就是所谓的回调方法咯  
            mOnClickListener.onClick(this);  
            return true;  
        }  
  
        return false;  
    }  
}  

注:setOnClickListener()方法只是给mOnClickListener赋值,当用户执行点击操作的时候,由于事件分发机制,最终会触发performClick()方法进而触发onClick()方法产生回调。注意一下思维:btn.setOnClickListener();并不是执行回调,而是给回调创造条件(performClick()方法中的if语句)。

流程分析:

MainActiviy.class执行onCreate(),执行到button.setOnClickListener(this); 假设此时mc为this所指代对象,即为MainActiviy.class的一个对象,将自己传入。
触发点击事件后系统执行View中的performClick(),该方法调用了mc.onClick(view v),此时把执行后结果返回给MainActivity.class,即由A类进行回调

button中onclick()的第二种写法:匿名类写法
//此时button为B(子)类,重写A(父)类View中的setOnClickListener()
button.setOnClickListener(new View.OnClickListener() {
@Override 
//注意:此时传入的参数是父类v,也就是说,会将执行结果返回给view
public void onClick(View v) { //做一些操作 doWork(); } 
});

//类似于B类中此方法

 public void setOnClickListener(Callback callback){
        String answer = "callback";
        /*B类方法中调用A类方法,这个A类方法要包括接口中的那个抽象方法,即doCallback(),并将所得结果作为参数传回A类中
         * 这里就是回调,实现内容在B类的回调中完成
         */
         callback.doCallback(answer);

View中代码:

//这里将OnClickListener赋值给了mOnClickListene
public void setOnClickListener(@Nullable OnClickListener l) 
{ if (!isClickable()) { 
setClickable(true); } 
getListenerInfo().mOnClickListener = l; }

public boolean performClick() { sendAccessibilityEvent(
AccessibilityEvent.TYPE_VIEW_CLICKED); 
ListenerInfo li = mListenerInfo; 
if (li != null && li.mOnClickListener != null) { playSoundEffect(
SoundEffectConstants.CLICK); li.mOnClickListener.onClick(this);
return true; } 
return false; }

触发点击事件后系统自动执行view中的perfromClick()方法,自然也就执行到了.mOnClickListener.onClick(view)######还是由View中onClick()调用

两种写法的差别在于:

第一种写法:A类为MainActivity.class,B类为View(Button继承View)
第二种写法:A类为View类,B类为Button类

回调的理解

回调其实是一种双向调用模式,也就说调用方在接口被调用时也会调用对方的接口
翻译翻译就是“实现了抽象类/接口 的实例实现了父类的提供的抽象方法后,将该方法交还给父类来处理”

实现方法交还给提供接口的父类处理!

不好的地方求指教!!!

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,014评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,796评论 3 386
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,484评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,830评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,946评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,114评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,182评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,927评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,369评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,678评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,832评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,533评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,166评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,885评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,128评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,659评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,738评论 2 351

推荐阅读更多精彩内容