詳解Webwork中Action 調(diào)用的方法
本文主要通過三個(gè)方面給大家介紹webwork action調(diào)用相關(guān)知識,三個(gè)方面分別是:
1.這部分框架類關(guān)系
2.Webwork 獲取和包裝 web 參數(shù)
3.DefaultActionProxyFactory、DefaultActionProxy、DefaultActionInvocation
一路走來,終于要開始 webwork 核心業(yè)務(wù)類的總結(jié),webwork 通過對客戶端傳遞的 web 參數(shù)重新包裝,進(jìn)行執(zhí)行業(yè)務(wù) Action 類,并反饋執(zhí)行結(jié)果,本篇源碼分析對應(yīng)下圖 WebWork 框架流轉(zhuǎn)圖中紅色框的地方。
1.這部分框架類關(guān)系
2.Webwork 獲取和包裝 web 參數(shù)
•每個(gè)Web 框架或多或少的對 Web 請求參數(shù)的包裝,用來拿來方便自己使用,當(dāng)然webwork 也不例外。
•Webwork 每次響應(yīng)請求的入口方法:
public void service(HttpServletRequest request, HttpServletResponse response) throws ServletException { try { if (encoding != null) { try { request.setCharacterEncoding(encoding); } catch (Exception localException) { } } if (locale != null) { response.setLocale(locale); } if (this.paramsWorkaroundEnabled) { request.getParameter("foo"); } request = wrapRequest(request); //封裝 request請求 serviceAction(request, response, getNameSpace(request), getActionName(request), getRequestMap(request), getParameterMap(request), getSessionMap(request), getApplicationMap()); } catch (IOException e) { String message = "Could not wrap servlet request with MultipartRequestWrapper!"; log.error(message, e); sendError(request, response, , new ServletException(message, e)); } }
•接受 request 、response 參數(shù),并對 request 參數(shù)進(jìn)行封裝,這次封裝主要是針對多媒體請求進(jìn)行的特殊處理,例如項(xiàng)目中的文件上傳請求,導(dǎo)出各種類型文件等...
•包裝完 request 之后,service 方法調(diào)用 ServletDispatche.serviceAction() 方法,并調(diào)用 getApplicationMap、getSessionMap、getRequestMap、 getParameterMap、getActionName、getNameSpace 6 個(gè)方法開始了Action 業(yè)務(wù)邏輯調(diào)用前的前戲。
•getNameSpace 方法用來獲得一個(gè)Action所屬的名稱空間,例如 : "/my/MyAction.action"則返回"/my",具體實(shí)現(xiàn)如下:
protected String getNameSpace(HttpServletRequest request){ String servletPath = request.getServletPath(); return getNamespaceFromServletPath(servletPath); } public static String getNamespaceFromServletPath(String servletPath){ servletPath = servletPath.substring(, servletPath.lastIndexOf("/")); return servletPath; }
•getActionName 返回請求的Action的名字,例如:"MyAction.action"則返回"MyAction",具體實(shí)現(xiàn)如下:
protected String getActionName(HttpServletRequest request){ String servletPath = (String)request.getAttribute("javax.servlet.include.servlet_path"); if (servletPath == null) { servletPath = request.getServletPath(); } return getActionName(servletPath); } protected String getActionName(String name){ int beginIdx = name.lastIndexOf("/"); int endIdx = name.lastIndexOf("."); return name.substring(beginIdx == - ? : beginIdx + , endIdx == - ? name.length() : endIdx); }
• getRequestMap 方法返回一個(gè)包含請求中所有屬性的Map,具體實(shí)現(xiàn)類是 RequestMap,具體代碼如下:
protected Map getRequestMap(HttpServletRequest request){ return new RequestMap(request); }
•getParameterMap 方法返回一個(gè)包含請求中所有參數(shù)的Map,具體代碼如下:
protected Map getParameterMap(HttpServletRequest request) throws IOException{ return request.getParameterMap(); }
•getSessionMap 方法返回一個(gè)包含 session 中所有屬性的 Map,具體實(shí)現(xiàn)類是 SessionMap,具體代碼如下:
protected Map getSessionMap(HttpServletRequest request){ return new SessionMap(request); }
•getApplicationMap 方法返回一個(gè)包含 Application 中所有屬性的Map,具體實(shí)現(xiàn)類 是ApplicationMap,具體代碼如下:
protected Map getApplicationMap(){ return new ApplicationMap(getServletContext()); }
•WebWork之所以要把request 的屬性、參數(shù),session 中的屬性,Application 中的屬性封裝成 Map,僅僅是為了自己使用方便。
public void serviceAction(HttpServletRequest request, HttpServletResponse response, String namespace, String actionName, Map requestMap, Map parameterMap, Map sessionMap, Map applicationMap) { HashMap extraContext = createContextMap(requestMap, parameterMap, sessionMap, applicationMap, request, response, getServletConfig()); extraContext.put("com.opensymphony.xwork.dispatcher.ServletDispatcher", this); OgnlValueStack stack = (OgnlValueStack) request.getAttribute("webwork.valueStack"); if (stack != null) { extraContext.put("com.opensymphony.xwork.util.OgnlValueStack.ValueStack", new OgnlValueStack(stack)); } try { ActionProxy proxy = ActionProxyFactory.getFactory().createActionProxy(namespace, actionName, extraContext); request.setAttribute("webwork.valueStack", proxy.getInvocation().getStack()); proxy.execute(); if (stack != null) { request.setAttribute("webwork.valueStack", stack); } } catch (ConfigurationException e) { log.error("Could not find action", e); sendError(request, response, 404, e); } catch (Exception e) { log.error("Could not execute action", e); sendError(request, response, 500, e); } }
•首先 ServiceAction 調(diào)用了createContextMap 創(chuàng)建Action 上下文(extraContext)。 它將JavaServlet 相關(guān)的對象進(jìn)行包裝,放入extraContext Map對象里。
•接著檢查 上一個(gè)請求中是否有可用的值堆棧,如果有就放入extraContext 這個(gè)Map 對象里,供本次請求使用 。
•ActionContext(com.opensymphony.xwork.ActionContext)是Action執(zhí)行時(shí)的上下文,上下文 可以看作是一個(gè)容器(其實(shí)我們這里的容器就是一個(gè)Map 而已),它存放的是Action 在執(zhí)行時(shí)需要用到的對象。
• ServletActionContext ( com.opensymphony.webwork. ServletActionContext),這個(gè)類直接繼承了ActionContext,它提供了直接與JavaServlet 相關(guān)象訪問的功能。
•OgnlValueStack主要的功能是通過表達(dá)式語言來存取對象的屬性。
3.DefaultActionProxyFactory、DefaultActionProxy、DefaultActionInvocation
前戲終于做完了,Action 調(diào)用的三兄弟要登場進(jìn)行最重要的操作了,就是下面這三句代碼,與Webwork 學(xué)習(xí)之路(五)請求跳轉(zhuǎn)前 xwork.xml 的讀取代碼有非常相似的寫法和設(shè)計(jì):
ActionProxy proxy = ActionProxyFactory.getFactory().createActionProxy(namespace, actionName, extraContext); request.setAttribute("webwork.valueStack", proxy.getInvocation().getStack());
proxy.execute(); •通過由前面獲得的namespace、actionName、extraContext 創(chuàng)建調(diào)用代理 ActonProxy 實(shí)例,這里也就是 DefaultActionProxy,之后調(diào)用 了 ActionProxy.execute 方法來執(zhí)行我們邏輯Action.execute。
•ActionProxy是一個(gè)接口,ActionProxyFactory則是一個(gè)抽象類,默認(rèn)情況下它們是通過 DefaultActionProxy和DefaultActionProxyFactory來完成操作的。
•在 ActionProxyFactory 中有一個(gè)靜態(tài)變量 factory ,它指向的是一個(gè) DefaultActionProxyFactory 實(shí)例,代碼如下:
static ActionProxyFactory factory = new DefaultActionProxyFactory(); public static void setFactory(ActionProxyFactory factory){ factory = factory; } public static ActionProxyFactory getFactory(){ return factory; }
• DefaultActionProxyFactory 的 createActionProxy 方法返回了 DefaultActionProxy 實(shí)例。
public ActionProxy createActionProxy(String namespace, String actionName, Map extraContext)throws Exception { setupConfigIfActionIsCommand(namespace, actionName); return new DefaultActionProxy(namespace, actionName, extraContext, true); } •DefaultActionProxy的構(gòu)造函數(shù) protected DefaultActionProxy(String namespace, String actionName, Map extraContext, boolean executeResult) throws Exception{ if (LOG.isDebugEnabled()) { LOG.debug("Creating an DefaultActionProxy for namespace " + namespace + " and action name " + actionName); } this.actionName = actionName; this.namespace = namespace; this.executeResult = executeResult; this.extraContext = extraContext; this.config = ConfigurationManager.getConfiguration().getRuntimeConfiguration().getActionConfig(namespace, actionName); if (this.config == null) { String message; String message; if ((namespace != null) && (namespace.trim().length() > 0)) { message = LocalizedTextUtil.findDefaultText("xwork.exception.missing-package-action", Locale.getDefault(), new String[] { namespace, actionName }); } else { message = LocalizedTextUtil.findDefaultText("xwork.exception.missing-action", Locale.getDefault(), new String[] { actionName }); } throw new ConfigurationException(message); } prepare(); }
•將傳入的名稱空間、 Action 的名字等參數(shù)賦予本地變量,接著通過 ConfigurationManager 獲得當(dāng)前請求的 Action 的配置信息[這里在5中已經(jīng)描述過]。接著調(diào)用自身的 prepare 方法創(chuàng)建一個(gè) ActionInvocation 對象賦予自身變量 invocation。在之后的 execute 方法中通過操縱invocation 來實(shí)現(xiàn)我們自己寫的Action 的調(diào)用。
protected void prepare() throws Exception { this.invocation = ActionProxyFactory.getFactory().createActionInvocation(this, this.extraContext); }
以上所示是針對Webwork中Action 調(diào)用 的相關(guān)知識,希望對大家有所幫助。
相關(guān)文章
javascript實(shí)現(xiàn)禁止復(fù)制網(wǎng)頁內(nèi)容匯總
本文給大家匯總介紹了幾種使用javascript和CSS實(shí)現(xiàn)禁止復(fù)制頁面內(nèi)容的方法,非常的實(shí)用,有需要的小伙伴可以參考下。2015-12-12JS和Canvas實(shí)現(xiàn)圖片的預(yù)覽壓縮和上傳功能
這篇文章主要介紹了JS和Canvas實(shí)現(xiàn)圖片的預(yù)覽壓縮和上傳功能,實(shí)現(xiàn)此功能大概有兩步,第一步用戶選擇需要上傳的圖片,第二步獲取圖片資源壓縮預(yù)覽上傳,具體實(shí)現(xiàn)代碼大家參考下本文2018-03-03js借助ActiveXObject實(shí)現(xiàn)創(chuàng)建文件
創(chuàng)建文件的方法有很多,在本文為大家詳細(xì)介紹下js中時(shí)如何實(shí)現(xiàn)的,感興趣的朋友不要錯(cuò)過了2013-09-09js+h5 canvas實(shí)現(xiàn)圖片驗(yàn)證碼
這篇文章主要為大家詳細(xì)介紹了js+h5 canvas實(shí)現(xiàn)圖片驗(yàn)證碼,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-10-10通過js動態(tài)操作table(新增,刪除相關(guān)列信息)
通過js動態(tài)操作table(新增,刪除相關(guān)列信息)的實(shí)現(xiàn)代碼,需要的朋友可以參考下2012-05-05