上篇說過,所有提交到web程序的url都被此UrlFilter攔截。攔截到請求后,UrlFilter則召集它的好多個得力干將Router 們, 詢問他們:“誰能處理此URL啊 ?”
這時一位叫做AMP的Router 首當其沖 說:“這個url交給我了”。這時filter就會把此url全權交給AMPRouter來辦,至于如何去處理,filter也不再過問,它覺得:“我把任務都交給你了,怎么解決是你的事”。
根據單一職責的原則,UrlFilter就負責上面情景中的分發url到Router中的差事,url如何分發交給Router處理。并且Router實際是一個接口,使用框架的用戶完全可以自己實現Router,這樣用戶可以自主定義的url分發的策略。另外呢,框架初始化的一些操作它也是 推脫不掉的,像根據unicorn-config.xml初始化系統中的Router和Action'。下面是具體的代碼:
@Override public void init(FilterConfig config) throws ServletException { application = config.getServletContext(); String loadPath = Thread.currentThread().getContextClassLoader().getResource("").getPath(); String classPath = loadPath.substring(1, loadPath.length()); ArrayList<String> actions = XMLReader.getNodeValues(classPath + "unicorn-config.xml", "actions"); this.initActions(actions); ArrayList<String> routers = XMLReader.getNodeValues(classPath + "unicorn-config.xml", "routers"); this.initRouters(routers); }
@Override public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException { HttpServletRequest request = (HttpServletRequest) servletRequest; String path = request.getContextPath(); String uri = request.getRequestURI(); String relativeUri = uri.substring(path.length(), uri.length()); this.request = request; this.session = request.getSession(); this.response = (HttpServletResponse) servletResponse; // 用戶自定義的Router優先級最高,url先通過用戶定義的 Iterator<IPathRouter> iterator = routerList.iterator(); while(iterator.hasNext()) { IPathRouter router = iterator.next(); if(router.route(relativeUri, this)) { return ; } } // 攔截不到的繼續訪問 filterChain.doFilter(servletRequest, servletResponse); }
其中Router類的初始化,Action類的初始化于這個類似:
private void initRouters(ArrayList<String> routers) { routerList = new ArrayList<IPathRouter>(); for (int i = 0; i < routers.size(); i++) { String routerName = routers.get(i); try { Class<?> clz = Class.forName(routers.get(i)); // 單例模式通過方法獲取對象實例 IPathRouter router = (IPathRouter) clz.newInstance(); routerList.add(router); } catch (ClassNotFoundException e) { e.printStackTrace(); } catch (IllegalAccessException e) { e.printStackTrace(); } catch (SecurityException e) { e.printStackTrace(); } catch (IllegalArgumentException e) { e.printStackTrace(); } catch (InstantiationException e) { e.printStackTrace(); } } // 最后把框架默認的Router加入進來 routerList.add(new AMPPathRouter()); }
其中unicorn-config.xml文件的編寫,拿其中我一個項目里的這個文件來舉例:
<?xml version="1.0" encoding="UTF-8" ?>
<config>
<routers>
<router class="com.mh.router.MySessionCheckRouter"></router>
</routers>
<actions>
<action class="com.mh.action.UserAction"></action>
<action class="com.mh.action.InformationAction"></action>
<action class="com.mh.action.UploadInformationIconAction"></action>
<action class="com.mh.action.TempPicAction"></action>
<action class="com.mh.action.MobileAction"></action>
</actions>
</config>
這里即定義了Action,也定義了自己的Router,并且從名稱上可以看出,這個SessionCheckRouter是要判斷所有提交到服務器的指定url的請求 是否已經登錄過,沒有登錄,可能會把此請求遣送會登錄頁。以及初始化所有的Action,在Router處理完請求,分發給action時,可以從filter里面去取。