浅谈回调接口

序言

有需求,才会去创造。
网络上有许多关于接口回调的文章,各有千秋,理解角度也分别不同。今天本人希望从需求角度,能让一些对于接口回调不了解,或者概念模糊的人理解一下。本来是想写一下回调的概念。但是我发现这样不利于理解,所以这里我就不赘述概念了,大家看这篇博文也不要带着固定的思维去理解,希望大家举一反三吧。这里我从一个需求的角度让大家理解一下回调的逻辑流程。

目的

java中接口回调随处可见,比如说各种监听,onClickListener,而最近比较热的Mvp框架,其中view层抽象接口,也属于接口回调,掌握他,你会发现,逻辑世界还是很神奇的。

应用场景

  1. 多线程之间数据同步问题
    更具象的说法,举个例子,我前段日子有个需求,我需要处理一些字段,但是这些字段里有一个值是北京时间,而北京时间是需要异步获取,这样我希望的就是获取到北京时间后,再处理。这里我就用了接口回调。 (ps:这里我将需求尽可能的简化,只是为了让大家理解回调流程,详细的需求是加密验证,这里就不详述了!否则有点本末倒置。)

    流程示范


编写功能模块

功能类A

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
package com.dong.test;

/**
*
* @author JDD 这是一个功能类 假设他用来处理一些字段
*
*/
public class ManageFields {
String str;
/**
* 用来处理字段的方法,而其中有一个字段是北京时间,需要开启异步线程获取时间后再进行一些逻辑处理
*/
public void doSomething(String string) {
//这里我需要获取北京时间,处理的话比如说拼接一段字段在北京时间前面
str = string;
//现在要获取北京时间,这个就是开启一个异步线程去获取,现在问题来了怎么获取它得到的异步时间并拼接起来的。这里就用到了回调
TimeUtil timeUtil = new TimeUtil();
timeUtil.getBjTime();
}
}

功能类B

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
package com.dong.test;

import java.io.IOException;
import java.net.MalformedURLException;
import java.net.URL;
import java.net.URLConnection;
import java.util.Date;

/**
*
* @author JDD 这个是个时间类 需要获取北京时间
*
*/
public class TimeUtil {

/**
* 开启异步线程去获取时间
*/
public void getBjTime() {
new Thread(new Runnable() {
@Override
public void run() {
URL url;
try {
url = new URL("http://www.baidu.com");
URLConnection uc = url.openConnection();// 生成连接对象
uc.connect(); // 发出连接
long ld = uc.getDate(); // 取得网站日期时间
Date date = new Date(ld); // 转换为标准时间对象
// 分别取得时间中的小时,分钟和秒,并输出
long bjTime = date.getTime();

} catch (MalformedURLException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} // 取得资源对象
catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}).start();
}

}

So 问题来了,该怎将获取的时间同步给A呢?这里大家可以发散一下思维,onclick事件,触发机制,也就是回调机制,它是完成点击(获取到时间后),执行回调函数 onClick(view v)(执行处理操作)。那这里我们将回调时间抽象成接口。

编写接口

调时间抽象成接口。

抽象接口

1
2
3
4
5
6
7
8
9
10
11
package com.dong.test;

/**
*
* @author JDD 时间回调接口 或者说就是监听时间获取
*
*/
public interface TimeListener {
//抽象方法,其中参数 就是获取到的时间
void returnTime(long bjTime);
}

改造模块实现回调

首先时间获取模块B中应该有一个实现了TimeListener对象的引用,这里我们去重载一下其构造函数

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
package com.dong.test;

import java.io.IOException;
import java.net.MalformedURLException;
import java.net.URL;
import java.net.URLConnection;
import java.util.Date;

/**
*
* @author JDD 这个是个时间类 需要获取北京时间
*
*/
public class TimeUtil {

TimeListener timeListener;
/**
* 这个是实现了TimeListener的对象的引用(其实就是ManageFileds的引用,这样就需要ManageFields类去实现接口)
* @param timeListener
*/
public TimeUtil(TimeListener timeListener){
this.timeListener=timeListener;
}

/**
* 开启异步线程去获取时间
*/
public void getBjTime() {
new Thread(new Runnable() {
@Override
public void run() {
URL url;
try {
url = new URL("http://www.baidu.com");
URLConnection uc = url.openConnection();// 生成连接对象
uc.connect(); // 发出连接
long ld = uc.getDate(); // 取得网站日期时间
Date date = new Date(ld); // 转换为标准时间对象
// 分别取得时间中的小时,分钟和秒,并输出
long bjTime = date.getTime();
//调用管理时间的接口
timeListener.returnTime(bjTime);

} catch (MalformedURLException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} // 取得资源对象
catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}).start();
}

}

管理字段的类就需要实现 TimeListener接口

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
package com.dong.test;


/**
*
* @author JDD 这是一个功能类 假设他用来处理一些字段
*
*/
public class ManageFields implements TimeListener {
String str;
/**
* 用来处理字段的方法,而其中有一个字段是北京时间,需要开启异步线程获取时间后再进行一些逻辑处理
*/
public void doSomething(String string) {
// 这里我需要获取北京时间,处理的话比如说拼接一段字段在北京时间前面
str = string;
// 现在要获取北京时间,这个就是开启一个异步线程去获取,并将this,也就是自己的引用传过去
TimeUtil timeUtil = new TimeUtil(this);
timeUtil.getBjTime();
}

/**
* 这里就是当异步线程获取到数据,调用timeListener.returnTime(bjTime)时,所回调的方法
*/
@Override
public void returnTime(long bjTime) {
// TODO Auto-generated method stub
System.out.println(str+bjTime);
}
}

测试代码

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
package com.dong.test;

public class TestCallBack {

public static void main(String[] args) {
// TODO Auto-generated method stub
ManageFields manageFields1=new ManageFields();
ManageFields manageFields2=new ManageFields();

manageFields1.doSomething("现在北京时间是:");
manageFields2.doSomething("Now BeiJingTime is:");


}

}

结果

Now BeiJingTime is:1459396779000
现在北京时间是:1459396780000

流程图

下面你有个大概思路了,我们看一下流程图(viso制作)

流程图

拓展

当然 管理字段类还可以写成大家熟知的匿名内部类形式

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
package com.dong.test;

/**
*
* @author JDD 这是一个功能类 假设他用来处理一些字段
*
*/
public class ManageFields implements TimeListener {
String str;
//匿名内部类实现
public void doSomething(String string) {
// 这里我需要获取北京时间,处理的话比如说拼接一段字段在北京时间前面
str = string;
// 现在要获取北京时间,这个就是开启一个异步线程去获取,并将this,也就是自己的引用传过去
TimeUtil timeUtil = new TimeUtil(new TimeListener() {

@Override
public void returnTime(long bjTime) {
// TODO Auto-generated method stub
System.out.println(str+bjTime);
}
});
timeUtil.getBjTime();
}
}

后话

当然,你要是了解Rxjava,或者观察者模式,抑或hanlder等,都是可以实现这种需求的,但是,总有一种场景,有一种工具更配的来。就跟工具箱里的扳手一样,你有很多扳手,但是也有不同的螺丝,匹配的来的工具不是更给力么!
然而回调有时候不是必须的,因为有一个问题就是callbackhell-回调地狱,我的认知就是由于各种回调嵌套,导致工程可读性,可塑性差。所以大家也要两面性的看带问题!

源码

接口回调样例源码