https://www.jianshu.com/p/38acf37b1e1f
1.委派模式簡介
在常用的23種設計模式中其實面沒有委派模式(delegate)的影子,但是在Spring中委派模式確實用的比較多的一種模式。
在spring中的體現:Spring MVC框架中的DispatcherServlet其實就用到了委派模式。
委派模式的作用: 基本作用就是負責任務的調用和分配任務,跟代理模式很像,可以看做是一種特殊情況下的靜態代理的全權代理,但是代理模式注重過程,而委派模式注重結果
2.委派模式的實例
先引用一張圖來介紹委派模式,下圖主要說明了 boss把任務給leader,而leader做了一個任務的分配和調度的工作,自己沒有做工作,而是把具體工作交給具體的執行者去做。
委派模式.png
下面直接給出實例:
執行的接口
/**
* @Project: spring
* @description: 執行的接口
* @author: sunkang
* @create: 2018-08-30 23:10
* @ModificationHistory who when What
**/
public interface IExcuter {
void excute(String command);
}
普通員工A
/**
* @Project: spring
* @description: 員工A執行某項命令
* @author: sunkang
* @create: 2018-08-30 23:10
* @ModificationHistory who when What
**/
public class ExcuterA implements IExcuter{
@Override
public void excute(String command) {
System.out.println("員工A 開始做"+command+"的工作");
}
}
普通員工B
/**
* @Project: spring
* @description: 員工B執行某項命令
* @author: sunkang
* @create: 2018-08-30 23:10
* @ModificationHistory who when What
**/
public class ExcuterB implements IExcuter{
@Override
public void excute(String command) {
System.out.println("員工B 開始做"+command+"的工作");
}
}
leader委派者
/**
* @Project: spring
* @description: leader 委派者 任務分發的作用
* @author: sunkang
* @create: 2018-08-30 23:11
* @ModificationHistory who when What
**/
public class Leader implements IExcuter {
private Map<String,IExcuter> targets = new HashMap<String,IExcuter>();
public Leader() {
targets.put("加密",new ExcuterA());
targets.put("登錄",new ExcuterB());
}
@Override
public void excute(String command) {
targets.get(command).excute(command);
}
}
boss類模擬調用測試
/**
* @Project: spring
* @description: boss 模擬客戶執行任務
* @author: sunkang
* @create: 2018-08-30 23:13
* @ModificationHistory who when What
**/
public class Boss
{
public static void main(String[] args) {
Leader leader = new Leader();
//看上去好像是我們的項目經理在幹活
//但實際幹活的人是普通員工
//這就是典型,幹活是我的,功勞是你的
leader.excute("登錄");
leader.excute("加密");
}
}
測試結果爲:
委派模式測試結果.png
3.委派模式的實現mvc的簡單實例
思考:
在springmvc訪問地址的url
與Controller層配置的url
是如何映射的
Controller層配置的url
如何跟具體的方法
映射的,參數
又是如何綁定
的
猜測:
url的地址可以根據request得到訪問url地址,配置的url地址可以根據配置註解得到,這兩者的url匹配上了說明映射成功了,除了url是不夠的,還需要一箇中間對象保存了url和method以及controller對象的信息,可以把這個中年的映射對象放入容器中,然後根據傳入的url從容器取出進行匹配,取出來之後就可以根據映射來完成方法的調用了。
下面就簡單實現一個mvc調用的例子
模擬controller層
/**
* @Project: spring
* @description: 模擬controller層
* @author: sunkang
* @create: 2018-09-03 22:20
* @ModificationHistory who when What
**/
public class MemberAction {
public void getMemberById(String mid){
}
}
模擬serlvert得到的請求如何轉發,交給具體的controller處理的
/**
* @Project: spring
* @description: selvelt的任務分發者 ,主要完成url的映射和調用
* @author: sunkang
* @create: 2018-09-03 22:21
* @ModificationHistory who when What
**/
public class SelvletDispatcher {
//這裏也可以用map 對象來保存Hanlder對象
private List<Handler> handlerMapping = new ArrayList<Handler>();
public SelvletDispatcher() {
//簡單實現一個controller的映射
try {
Class clazz = MemberAction.class;
handlerMapping.add(new Handler()
.setController(clazz.newInstance())
.setMethod(clazz.getMethod("getMemberById",new Class[]{String.class}))
.setUrl("/web/getMemberById.json")
);
} catch (Exception e) {
e.printStackTrace();
}
}
private void doService(HttpServletRequest request, HttpServletResponse response){
doDispatch(request,response);
}
/**
* 請求的分發工作
* @param request
* @param response
*/
private void doDispatch(HttpServletRequest request, HttpServletResponse response) {
//1.獲取用戶請求的url
String uri = request.getRequestURI();
Handler handler =null;
////2、根據uri 去handlerMapping找到對應的hanler
for(Handler h :handlerMapping){
if(uri.equals(h.getUrl())){
handler = h;
break;
}
}
//3.將具體的任務分發給Method(通過反射去調用其對應的方法)
Object obj = null;
try {
obj = handler.getMethod().invoke(handler.getController(),request.getParameter("mid"));
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (InvocationTargetException e) {
e.printStackTrace();
}
//4、獲取到Method執行的結果,通過Response返回出去
// response.getWriter().write();
}
/**
* 具體的hanlder對象
*/
class Handler{
//controller對象
private Object controller;
//controller對象映射的方法
private String url;
//ulr對應的方法
private Method method;
public Object getController() {
return controller;
}
public Handler setController(Object controller) {
this.controller = controller;
return this;
}
public String getUrl() {
return url;
}
public Handler setUrl(String url) {
this.url = url;
return this;
}
public Method getMethod() {
return method;
}
public Handler setMethod(Method method) {
this.method = method;
return this;
}
}
}