日常开发中接口回调机制无处不在,刚开始用时却总是晕晕乎乎,网上也有很多相关的文章介绍,但总是没有看得太明白,今天端午假期正好花时间来总结一下,我们按如下顺序介绍
一、什么是接口回调
在应用开发中,接口回调机制是一种常用的设计手段,也可以说是一种处理问题的模型,类之间,模块之间,都有一定的调用关系,一般来说,可以把使用某一接口的类创建的对象的引用赋给该接口声明的接口变量,那么该接口变量就可以调用被类实现的接口方法,实际上,当接口变量调用被类实现的接口中的方法时,就是通知相应的对象调用接口的方法,这个调用的过程称为接口的回调,是不是很晕乎,无奈,理论总是很抽象,后面我们会通过具体的实例来验证,看完实例大家可以再回来体会一下,看看是不是这个理
二、回调的应用场景
既然接口回调是开发应用中处理问题的核心手段,那我们来看看它到底有什么应用场景,也就是说具体在什么情况下,我们要想到使用接口回调来解决问题,回调一般用于分层间的互相协作,上层将本层函数安装在下层,这个函数就是回调,而下层在一定条件下触发回调,例如,作为一个驱动,是底层,它在收到一个数据时,除了完成本层的处理工作外,还将进行回调,将数据交给上层做进一步的处理,另外我们在封装中也经常用到,还有就是 View 的点击事件其实也是使用回调的原理
三、Java 中实现接口回调
原理:首先创建一个对象,然后再创建一个控制器对象,将回调对象需要被调用的方法告诉控制器对象,控制器对象负责检查某个场景是否出现或某个条件是否满足,当满足时,自动调用回调对象方法
例如程序员 A 和 B 之间需要协作,A 告诉 B 这个任务中间的某个环节需要 B 来完成,并且完成后告诉 A,这时候程序员 A 就需要告诉 B 一个联系方式,使 B 完成时来通知自己,这个场景就可以使用回调,
示例代码如下:
1.首先我们先来创建一个回调接口
这里主要是得让程序员 B 干完活如何找到程序员 A
/**
* 回调接口
* 此接口为联系的方式,程序员A必须要实现此接口
* Created by qiudengjiao on 2017/5/27.
*/
public interface Callback {
void event(String result);
}
/**
* 层序员 A
* 层序员 A 是作为上层应用身份出现的,下层应用(程序员 B)是不知道
* 有哪些方法,因此它想被下层应用(程序员 B)调用必须实现此接口
* Created by qiudengjiao on 2017/5/27.
*/
public class ProgrammerA implements Callback {
//B程序员对象引用
private ProgrammerB programmerB;
//在构造方法中国持有B程序员的引用
public ProgrammerA(ProgrammerB programmerB) {
this.programmerB = programmerB;
}
/**
* 程序员A通过这个方法告诉程序员B任务
*/
public void doEvent(final String event) {
//这里用一个线程就是异步,
new Thread(new Runnable() {
@Override
public void run() {
//程序员A调用程序员B中的方法,在这里注册回调接口
programmerB.doWork(ProgrammerA.this, event);
}
}).start();
}
/**
* 程序员B完成任务后调用此方法告诉A,也就是程序员A的回调方法
*
* @param result
*/
@Override
public void event(String result) {
Log.e("程序员B告诉程序员A:", result);
}
}
/**
* 程序员 B
* 必须要注意,这是一个底层类,底层是不了解上层服务的
* Created by qiudengjiao on 2017/5/27.
*/
public class ProgrammerB {
public void doWork(Callback callback, String event) {
Log.e("程序员A告诉程序员B需要干的事:", event);
Log.e("程序员B:", "干活.....");
String result = "完成工作";
//程序员B在这里调用A回调方法,告诉完成任务
callback.event(result);
}
}
/**
* Callback 测试
*/
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
init();
}
private void init() {
ProgrammerB programmerB = new ProgrammerB();
ProgrammerA programmerA = new ProgrammerA(programmerB);
programmerA.doEvent("编写一个列表界面");
}
}
运行结果截图:
到这里我觉得大家对回调也就有了基本的认识,接下来就是需要去亲自动手实践一下,自己体会体会,毕竟实践才是最好的老师
四、接口回调在 Android 中的应用
1.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);
}
2.回调函数
/**
* Callback 测试
*/
public class MainActivity extends AppCompatActivity implements View.OnClickListener{
private Button button;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
initView();
init();
}
private void initView() {
button = (Button) findViewById(R.id.button);
button.setOnClickListener(this);
}
private void init() {
ProgrammerB programmerB = new ProgrammerB();
ProgrammerA programmerA = new ProgrammerA(programmerB);
programmerA.doEvent("编写一个列表界面");
}
/**
* 用户点击Button时调用的回调函数
* @param v
*/
@Override
public void onClick(View v) {
Toast.makeText(this,"onClick",Toast.LENGTH_SHORT).show();
}
}
3.View 类的 setOnClickListener 方法
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;
}
/**
* Call this view's OnClickListener, if it is defined.
*
* @return True there was an assigned OnClickListener that was called, false
* otherwise is returned.
*/
public boolean performClick() {
sendAccessibilityEvent(AccessibilityEvent.TYPE_VIEW_CLICKED);
if (mOnClickListener != null) {
playSoundEffect(SoundEffectConstants.CLICK);
//回调方法
mOnClickListener.onClick(this);
return true;
}
return false;
}
}
五、总结
1. 必须明确回调函数出现的原因,也就是使用的场景
2.还有就是在工作中,项目中可能会把一部分功能交给被人或别的公司来做,实现个性化定制功能,这里别人具体去怎么实现,我们便无法得知,其实我们也没必要去管,我们只需要做的就是定义好相关接口,这一设计允许了底层代码调用高层定义的子程序,增强了程序的灵活性,这才是回调的真正原因
3.在封装过程中,上层模块封装时,很难知道下层模块是如何实现的,因此,上层模块只需要定义好自己需要但不能预料的接口(也就是回调接口),当下层模块调用上层模块时,根据当前需要的实现回调接口,并通过注册或参数方式传入上层模块即可,这样就实现了下层调用上层,并且上层还能根据传入的引用来调用下层的具体实现,将程序的灵活性大大的增加