一、原理
当有两个或两个以上对象有机会处理某个请求时,可以将这些对象添加到一个对象链表,然后对链表中的对象依次遍历,如果某个对象可以处理请求则遍历结束,该请求被这个对象处理消耗掉;否则,如果该对象无法处理请求,就请求链表中的下一个对象,以此类推,这就是责任链模式。 责任链模式可以将请求发送者和处理者解耦,请求发送者只和链表头发生联系,这样也便于链表的排序和增删,当然因为循环调用代码调试以及排错可能也麻烦些。 android中用到很多责任链模式的地方,比如view事件分发、有序广播、以及okhttp中的拦截器。
1、view事件分发
android中MotionEvent事件下发到Activity后,Activity的dispatchTouchEvent方法调用了PhoneWindow,PhoneWindow的superDispatchTouchEvent方法调用了superDispatchTouchEvent方法,最终走到了ViewGroup的dispatchTouchEvent方法。在android事件分发机制一文中介绍了ViewGroup的onInterceptTouchEvent方法判断ViewGroup自身是否拦截事件,如果拦截则ViewGroup自身通过onTouchEvent方法消耗事件;否则调用子View的dispatchTouchEvent方法;如果子view和ViewGroup都不处理事件那么最终Activity的onTouchEvent方法被调用。
2、okhttp拦截器
在okhttp执行与拦截器一文中提到实际执行网络请求任务时在RealCall.java的getResponseWithInterceptorChain()方法中会添加lient.interceptors、retryAndFollowUpInterceptor、BridgeInterceptor、CacheInterceptor、ConnectInterceptor、networkInterceptors、CallServerInterceptor这几个拦截器到集合中,这些拦截器都实现了intercept接口去处理自身事务,然后调用拦截器链表接口方法proceed去触发后面的拦截器。
二、实践
假设有一道试题,可能是英语题、数学题、语文题中的一种,需要相应科目的老师去批改,这就可以使用责任链模式来处理。 抽象老师类
public abstract class Teacher {
protected Teacher successor;
public final void handleRequest(String test){
if (TextUtils.isEmpty(test)) {
LogUtil.d("error,test can not be null");
return;
}
if (test.equals(getSubjects())) {
handle(test);
}else if(successor!=null){
successor.handleRequest(test);
}else {
LogUtil.d("sorry, the test is so difficult that no teacher can review it");
}
}
protected abstract String getSubjects();
protected abstract void handle(String test);
}
语文老师类
public class ChineseTeacher extends Teacher {
@Override
protected String getSubjects() {
return "Chinese";
}
@Override
protected void handle(String test) {
LogUtil.d("ChineseTeacher is reviewing"+test);
}
}
英语老师类
public class EnglishTeacher extends Teacher {
@Override
protected String getSubjects() {
return "English";
}
@Override
protected void handle(String test) {
LogUtil.d("EnglishTeacher is reviewing"+test);
}
}
数学老师类
public class MathTeacher extends Teacher {
@Override
protected String getSubjects() {
return "Math";
}
@Override
protected void handle(String test) {
LogUtil.d("MathTeacher is reviewing"+test);
}
}
请求任务如下:
ChineseTeacher chineseTeacher = new ChineseTeacher();
EnglishTeacher englishTeacher = new EnglishTeacher();
MathTeacher mathTeacher = new MathTeacher();
chineseTeacher.successor = englishTeacher;
englishTeacher.successor = mathTeacher;
chineseTeacher.handleRequest("Math");
log打印如下:
MathTeacher is reviewingMath
|