Android回调机制浅谈

2017-06-09  本文已影响0人  beautymo

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

经典回调方式

下面举一个小例子

/*定义一个接口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();
    }
}

流程分析:
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类

回调的理解

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

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

不好的地方求指教!!!

上一篇下一篇

猜你喜欢

热点阅读