java开发之——[接口回调]

一、回调的含义和用途

1. 什么是回调?

一般来说,模块之间都存在一定的调用关系,从调用方式上看,可以分为三类:同步调用、异步调用和回调。同步调用是一种阻塞式调用,即在函数A的函数体里通过书写函数B的函数名来调用之,使内存中对应函数B的代码得以执行。异步调用是一种类似消息或事件的机制解决了同步阻塞的问题,例如 A通知 B后,他们各走各的路,互不影响,不用像同步调用那样, A通知 B后,非得等到 B走完后, A才继续走 。回调是一种双向的调用模式,也就是说,被调用的接口被调用时也会调用对方的接口,例如A要调用B,B在执行完又要调用A。

2. 回调的用处

回调一般用于层间协作,上层将本层函数安装在下层,这个函数就是回调,而下层在一定条件下触发回调。例如作为一个驱动,是一个底层,他在收到一个数据时,除了完成本层的处理工作外,还将进行回调,将这个数据交给上层应用层来做进一步处理,这在分层的数据通信中很普遍。

二、Java 实现接口回调

在C/C++中,要实现回调函数,被调用函数要告诉调用者自己的指针地址。但是Java没有指针地址,不能传递方法的地址,一般采用接口回调的方法来实现:把实现某一接口的类创建的对象的引用赋给该接口声明的接口变量,那么该接口变量就可以调用被调用类实现的接口的方法。

原理:首先创建一个回调对象,然后再创建一个控制器对象,将回调对象需要被调用的方法告诉控制器对象,控制器对象负责检查某个场景是否出现或某个条件是否满足,当满足时,自动调用回调对象的方法。

例如老板A对员工B说,我现在交给你一个任务,并且我把我的电话号码给你,你一旦完成任务就给我打电话。

详细的代码如下:

  1. 创建一个回调接口:

    1  public interface CallBack {
    2      public void doEvent();
    3  }

    2.创建回调接口的实现类,此例中,员工干完活后还要干什么事情是老板说了算的。

    1  public class Boss implements CallBack {
    2      public void doEvent() {
    3          System.out.println("打电话给老板,告知已经完成工作了");
    4      }
    5  }

    3.创建控制类,也就是本例中的员工对象,他要持有老板的地址(即回调接口)

     1 public class Employee {
     2      CallBack callBack;
     3      public Employee(CallBack callBack) {
     4          this.callBack=callBack;
     5      }
     6      public void doWork() {
     7          System.out.println("玩命干活中....");
     8          callBack.doEvent();
     9      }
    10  }

    4.测试类

    1 public class TestMain {
    2      public static void main(String[] args) {
    3          //创建控制器对象,将提供给他的回调对象传入
    4          Employee employee=new Employee(new Boss());
    5          //启动控制器对象运行
    6          employee.doWork();
    7      }
    8  }

    三、Android中的接口回调

    在 Android 中回调机制被大量的使用。比如,在 Activity 中定义了很多生命周期的不同状态要调用的方法,这些方法都是空实现,系统框架要调用,用户也要调用来实现。

    举个简单的例子就是Button的点击响应事件实现机制

    1 button.setOnClickListener(new OnClickListener() {
    2     @Override
    3     public void onClick(View v) {
    4 
    5     }
    6 });

    OnClickListener 就是 Android 系统所约好的接口,然后在我们写的应用程序中传入回调对象,这样就可以达到接口统一,实现不同的效果。这种实现机制类似于下面的代码:

     1 public class A {
     2     public void setOnClickListener(OnClickListener onClickListener) {
     3         onClickListener.onClick();
     4     }
     5 
     6     public interface OnClickListener {
     7         public void onClick();
     8     }
     9 }
    10 
    11 
    12 
    13 public class B {
    14     public static void main(String[] args) {
    15         A a=new A();
    16         a.setOnClickListener(new OnClickListener() {
    17             public void onClick() {
    18                 // TODO 自动生成的方法存根
    19 
    20             }
    21         });
    22     }
    23 }

    其中A相当于Button,a即button按钮,B类相当于View。

 

 

形象比喻:

你到一个商店买东西,刚好你要的东西没有货,于是你在店员那里留下了你的电话,过了几天店里有货了,店员就打了你的电话,然后你接到电话后就到店里去取了货。在这个例子里,你的电话号码就叫回调函数,你把电话留给店员就叫登记回调函数,店里后来有货了叫做触发了回调关联的事件,店员给你打电话叫做调用回调函数,你到店里去取货叫做响应回调事件。(来自知乎)

回调的好处:

降低代码的耦合性,使代码更灵活、简洁

步骤一 :定义回调接口

1 /**
2  * Created by pengkv on 15/10/22.
3  * 网络请求回调接口
4  */
5 public interface HttpCallBackListener {
6     void onFinish(String respose);
7 
8     void onError(Exception e);
9 }

步骤二:定义回调函数(将接口作为参数)

 1 /**
 2  * Created by pengkv on 15/10/22.
 3  * 网络请求工具类
 4  */
 5 public class HttpUtil {
 6 
 7     public static void requestData(final String urlStr, final HttpCallBackListener listener) {
 8         new Thread(new Runnable() {
 9             @Override
10             public void run() {
11                 HttpURLConnection connection = null;
12                 try {
13                     URL url = new URL(urlStr);
14                     connection = (HttpURLConnection) url.openConnection();
15                     connection.setRequestMethod("GET");
16                     connection.setConnectTimeout(8000);
17                     connection.setReadTimeout(8000);
18                     connection.setDoInput(true);
19                     connection.setDoOutput(true);
20                     InputStream in = connection.getInputStream();
21                     BufferedReader br = new BufferedReader(new InputStreamReader(in));
22                     StringBuilder sb = new StringBuilder();
23                     String line;
24                     while ((line = br.readLine()) != null) {
25                         sb.append(line);
26                     }
27                     if (listener != null) {
28                         //回调onFinish方法
29                         listener.onFinish(sb.toString());
30                     }
31                 } catch (Exception e) {
32                     if (listener != null) {
33                         //回调onError方法
34                         listener.onError(e);
35                     }
36                 } finally {
37                     if (connection != null) {
38                         connection.disconnect();
39                     }
40                 }
41             }
42         }).start();
43     }
44 }

步骤三:使用回调方法一

 1 public class MainActivity extends AppCompatActivity {
 2 
 3     @Override
 4     protected void onCreate(Bundle savedInstanceState) {
 5         super.onCreate(savedInstanceState);
 6         setContentView(R.layout.activity_main);
 7 
 8         HttpUtil.requestData("请求的网址", new HttpCallBackListener() {
 9             @Override
10             public void onFinish(String respose) {
11                 //处理请求
12             }
13 
14             @Override
15             public void onError(Exception e) {
16                 //处理异常
17             }
18         });
19     }
20 
21 }

步骤三:使用回调方法二

 1 public class MainActivity extends AppCompatActivity implements HttpCallBackListener {
 2 
 3     @Override
 4     protected void onCreate(Bundle savedInstanceState) {
 5         super.onCreate(savedInstanceState);
 6         setContentView(R.layout.activity_main);
 7 
 8         HttpUtil.requestData("请求的网址", this);
 9     }
10 
11 
12     @Override
13     public void onFinish(String respose) {
14         //处理请求
15     }
16 
17     @Override
18     public void onError(Exception e) {
19         //处理异常
20     }
21 
22 }

 

posted @ 2017-01-19 11:40  mingruqi  阅读(2766)  评论(0编辑  收藏  举报