详解Webwork中Action 调用的方法

从三方面介绍webwork action调用相关知识:

1.Webwork 获取和包装 web 参数

2.这部分框架类关系

3.DefaultActionProxyFactory、DefaultActionProxy、DefaultActionInvocation

终于要开始 webwork 核心业务类的总结,webwork 通过对客户端传递的 web 参数重新包装,进行执行业务 Action 类,并反馈执行结果,本篇源码分析对应下图 WebWork 框架流转图中红色框的地方。

1.这部分框架类关系

2.Webwork 获取和包装 web 参数

•每个Web 框架或多或少的对 Web 请求参数的包装,用来拿来方便自己使用,当然webwork 也不例外。 •Webwork 每次响应请求的入口方法:

  1. public void service(HttpServletRequest request, HttpServletResponse response) throws ServletException {
  2. try {
  3. if (encoding != null) {
  4. try {
  5. request.setCharacterEncoding(encoding);
  6. } catch (Exception localException) {
  7. }
  8. }
  9. if (locale != null) {
  10. response.setLocale(locale);
  11. }
  12. if (this.paramsWorkaroundEnabled) {
  13. request.getParameter("foo");
  14. }
  15. request = wrapRequest(request); //封装 request请求
  16. serviceAction(request, response, getNameSpace(request), getActionName(request), getRequestMap(request), getParameterMap(request), getSessionMap(request), getApplicationMap());
  17. } catch (IOException e) {
  18. String message = "Could not wrap servlet request with MultipartRequestWrapper!";
  19. log.error(message, e);
  20. sendError(request, response, , new ServletException(message, e));
  21. }
  22. }

•接受 request 、response 参数,并对 request 参数进行封装,这次封装主要是针对多媒体请求进行的特殊处理,例如项目中的文件上传请求,导出各种类型文件等...

•包装完 request 之后,service 方法调用 ServletDispatche.serviceAction() 方法,并调用 getApplicationMap、getSessionMap、getRequestMap、 getParameterMap、getActionName、getNameSpace 6 个方法开始了Action 业务逻辑调用前的前戏。

•getNameSpace 方法用来获得一个Action所属的名称空间,例如 : "/my/MyAction.action"则返回"/my",具体实现如下:

  1. protected String getNameSpace(HttpServletRequest request){
  2. String servletPath = request.getServletPath();
  3. return getNamespaceFromServletPath(servletPath);
  4. }
  5. public static String getNamespaceFromServletPath(String servletPath){
  6. servletPath = servletPath.substring(, servletPath.lastIndexOf("/"));
  7. return servletPath;
  8. }

•getActionName 返回请求的Action的名字,例如:"MyAction.action"则返回"MyAction",具体实现如下:

  1. protected String getActionName(HttpServletRequest request){
  2.  
  3. String servletPath = (String)request.getAttribute("javax.servlet.include.servlet_path");
  4.  
  5. if (servletPath == null) {
  6.  
  7. servletPath = request.getServletPath();
  8.  
  9. }
  10.  
  11. return getActionName(servletPath);
  12.  
  13. }
  14.  
  15. protected String getActionName(String name){
  16.  
  17. int beginIdx = name.lastIndexOf("/");
  18.  
  19. int endIdx = name.lastIndexOf(".");
  20.  
  21. return name.substring(beginIdx == - ? : beginIdx + , endIdx == - ? name.length() : endIdx);
  22.  
  23. }
• getRequestMap 方法返回一个包含请求中所有属性的Map,具体实现类是 RequestMap,具体代码如下:
  1. protected Map getRequestMap(HttpServletRequest request){
  2. return new RequestMap(request);
  3. }

•getParameterMap 方法返回一个包含请求中所有参数的Map,具体代码如下:

  1. protected Map getParameterMap(HttpServletRequest request) throws IOException{
  2. return request.getParameterMap();
  3. }

•getSessionMap 方法返回一个包含 session 中所有属性的 Map,具体实现类是 SessionMap,具体代码如下:

  1. protected Map getSessionMap(HttpServletRequest request){
  2. return new SessionMap(request);
  3. }

•getApplicationMap 方法返回一个包含 Application 中所有属性的Map,具体实现类 是ApplicationMap,具体代码如下:

  1. protected Map getApplicationMap(){
  2. return new ApplicationMap(getServletContext());
  3. }

•WebWork之所以要把request 的属性、参数,session 中的属性,Application 中的属性封装成 Map,仅仅是为了自己使用方便。

  1. public void serviceAction(HttpServletRequest request, HttpServletResponse response, String namespace, String actionName, Map requestMap, Map parameterMap, Map sessionMap, Map applicationMap) {
  2. HashMap extraContext = createContextMap(requestMap, parameterMap, sessionMap, applicationMap, request, response, getServletConfig());
  3. extraContext.put("com.opensymphony.xwork.dispatcher.ServletDispatcher", this);
  4. OgnlValueStack stack = (OgnlValueStack) request.getAttribute("webwork.valueStack");
  5. if (stack != null) {
  6. extraContext.put("com.opensymphony.xwork.util.OgnlValueStack.ValueStack", new OgnlValueStack(stack));
  7. }
  8. try {
  9. ActionProxy proxy = ActionProxyFactory.getFactory().createActionProxy(namespace, actionName, extraContext);
  10. request.setAttribute("webwork.valueStack", proxy.getInvocation().getStack());
  11. proxy.execute();
  12. if (stack != null) {
  13. request.setAttribute("webwork.valueStack", stack);
  14. }
  15. } catch (ConfigurationException e) {
  16. log.error("Could not find action", e);
  17. sendError(request, response, 404, e);
  18. } catch (Exception e) {
  19. log.error("Could not execute action", e);
  20. sendError(request, response, 500, e);
  21. }
  22. }

•首先 ServiceAction 调用了createContextMap 创建Action 上下文(extraContext)。 它将JavaServlet 相关的对象进行包装,放入extraContext Map对象里。

•接着检查 上一个请求中是否有可用的值堆栈,如果有就放入extraContext 这个Map 对象里,供本次请求使用 。

•ActionContext(com.opensymphony.xwork.ActionContext)是Action执行时的上下文,上下文 可以看作是一个容器(其实我们这里的容器就是一个Map 而已),它存放的是Action 在执行时需要用到的对象。

• ServletActionContext ( com.opensymphony.webwork. ServletActionContext),这个类直接继承了ActionContext,它提供了直接与JavaServlet 相关象访问的功能。

•OgnlValueStack主要的功能是通过表达式语言来存取对象的属性。

3.DefaultActionProxyFactory、DefaultActionProxy、DefaultActionInvocation

前戏终于做完了,Action 调用的三兄弟要登场进行最重要的操作了,就是下面这三句代码,与Webwork 学习之路(五)请

ActionProxy proxy = ActionProxyFactory.getFactory().createActionProxy(namespace, actionName, extraContext);
request.setAttribute("webwork.valueStack", proxy.getInvocation().getStack());
 

proxy.execute(); •通过由前面获得的namespace、actionName、extraContext 创建调用代理 ActonProxy 实例,这里也就是 DefaultActionProxy,之后调用 了 ActionProxy.execute 方法来执行我们逻辑Action.execute。

•ActionProxy是一个接口,ActionProxyFactory则是一个抽象类,默认情况下它们是通过 DefaultActionProxy和DefaultActionProxyFactory来完成操作的。

•在 ActionProxyFactory 中有一个静态变量 factory ,它指向的是一个 DefaultActionProxyFactory 实例,代码如下:

  1. static ActionProxyFactory factory = new DefaultActionProxyFactory();
  2. public static void setFactory(ActionProxyFactory factory){
  3. factory = factory;
  4. }
  5. public static ActionProxyFactory getFactory(){
  6. return factory;
  7. }

• DefaultActionProxyFactory 的 createActionProxy 方法返回了 DefaultActionProxy 实例。

  1. public ActionProxy createActionProxy(String namespace, String actionName, Map extraContext)throws Exception {
  2. setupConfigIfActionIsCommand(namespace, actionName);
  3. return new DefaultActionProxy(namespace, actionName, extraContext, true);
  4. } DefaultActionProxy的构造函数
  5. protected DefaultActionProxy(String namespace, String actionName, Map extraContext, boolean executeResult) throws Exception{
  6. if (LOG.isDebugEnabled()) {
  7. LOG.debug("Creating an DefaultActionProxy for namespace " + namespace + " and action name " + actionName);
  8. }
  9. this.actionName = actionName;
  10. this.namespace = namespace;
  11. this.executeResult = executeResult;
  12. this.extraContext = extraContext;
  13. this.config = ConfigurationManager.getConfiguration().getRuntimeConfiguration().getActionConfig(namespace, actionName);
  14. if (this.config == null)
  15. {
  16. String message;
  17. String message;
  18. if ((namespace != null) && (namespace.trim().length() > 0)) {
  19. message = LocalizedTextUtil.findDefaultText("xwork.exception.missing-package-action", Locale.getDefault(), new String[] {
  20. namespace, actionName });
  21. } else {
  22. message = LocalizedTextUtil.findDefaultText("xwork.exception.missing-action", Locale.getDefault(), new String[] {
  23. actionName });
  24. }
  25. throw new ConfigurationException(message);
  26. }
  27. prepare();
  28. }

•将传入的名称空间、 Action 的名字等参数赋予本地变量,接着通过 ConfigurationManager 获得当前请求的 Action 的配置信息[这里在5中已经描述过]。接着调用自身的 prepare 方法创建一个 ActionInvocation 对象赋予自身变量 invocation。在之后的 execute 方法中通过操纵invocation 来实现我们自己写的Action 的调用。

  1. protected void prepare() throws Exception {
  2. this.invocation = ActionProxyFactory.getFactory().createActionInvocation(this, this.extraContext);
  3. }

详解Webwork中Action 调用的方法的更多相关文章

  1. [原创]java WEB学习笔记55:Struts2学习之路---详解struts2 中 Action,如何访问web 资源,解耦方式(使用 ActionContext,实现 XxxAware 接口),耦合方式(通过ServletActionContext,通过实现 ServletRequestAware, ServletContextAware 等接口的方式)

    本博客的目的:①总结自己的学习过程,相当于学习笔记 ②将自己的经验分享给大家,相互学习,互相交流,不可商用 内容难免出现问题,欢迎指正,交流,探讨,可以留言,也可以通过以下方式联系. 本人互联网技术爱 ...

  2. 详解Java中的Object.getClass()方法

    详解Object.getClass()方法,这个方法的返回值是Class类型,Class c = obj.getClass(); 通过对象c,我们可以获取该对象的所有成员方法,每个成员方法都是一个Me ...

  3. 详解JavaScript中的arc的方法

    今天说说JavaScript在网页中画圆的函数arc! 一.arc所需要的参数设置 1 arc(x, y, radius, startAngle, endAngle, counterclockwise ...

  4. 详解 JS 中 new 调用函数原理

    JavaScript 中经常使用构造函数创建对象(通过 new 操作符调用一个函数),那在使用 new 调用一个函数的时候到底发生了什么?先看几个例子,再解释背后发生了什么. 1)看三个例子 1.1 ...

  5. 详解Java中的clone方法

    详解Java中的clone方法 参考:http://blog.csdn.net/zhangjg_blog/article/details/18369201/ 所谓的复制对象,首先要分配一个和源对象同样 ...

  6. 举例详解Python中的split()函数的使用方法

    这篇文章主要介绍了举例详解Python中的split()函数的使用方法,split()函数的使用是Python学习当中的基础知识,通常用于将字符串切片并转换为列表,需要的朋友可以参考下   函数:sp ...

  7. jQuery:详解jQuery中的事件(二)

    上一篇讲到jQuery中的事件,深入学习了加载DOM和事件绑定的相关知识,这篇主要深入讨论jQuery事件中的合成事件.事件冒泡和事件移除等内容. 接上篇jQuery:详解jQuery中的事件(一) ...

  8. 【转】详解C#中的反射

    原帖链接点这里:详解C#中的反射   反射(Reflection) 2008年01月02日 星期三 11:21 两个现实中的例子: 1.B超:大家体检的时候大概都做过B超吧,B超可以透过肚皮探测到你内 ...

  9. MySQL关闭过程详解和安全关闭MySQL的方法

    MySQL关闭过程详解和安全关闭MySQL的方法 www.hongkevip.com 时间: -- : 阅读: 整理: 红客VIP 分享到: 红客VIP(http://www.hongkevip.co ...

随机推荐

  1. 《JAVA与模式》之组合模式

    定义(GoF<设计模式>): 将对象组合成树形结构以表示“部分整体”的层次结构.组合模式使得用户对单个对象和使用具有一致性. 及角色: 1.Component 是组合中的对象声明接口,在适 ...

  2. R语言-神经网络包RSNNS

    code{white-space: pre;} pre:not([class]) { background-color: white; }if (window.hljs && docu ...

  3. linux下log4j乱码解决

    使用log4j的时候,在WIN系统的时候正常显示中文,但是发布到linux系统的时候中文就显示成乱码了 由于log4j配置文件中没有设置编码格式(encoding),所以log4j就使用系统默认编码. ...

  4. 使用WebMatrix发布网站

    使用WebMatrix发布网站 WebMatrix 简介: Microsoft WebMatrix 是微软最新的 Web 开发工具,它包含了构建网站所需要的一切元素.您可以从开源 Web 项目或者内置 ...

  5. Microsoft Dynamics CRM 前瑞开发

    做CRM开发最大的感受就是其前瑞开发过程中,调试起来比较麻烦,需要做一些断点还要配制一些浏览器设置,对新手来说比较困难.还有就是对REST调试,经常为了调试一个正确的结果而花费大量的时间.现在推荐一个 ...

  6. [ html canvas globalCompositeOperation ] canvas绘图属性 设置合成图像如何显示 属性演示

    <!DOCTYPE html> <html lang='zh-cn'> <head> <title>Insert you title</title ...

  7. 上传Android代码到Jcenter(解决了字符映射的问题)

    请先阅读:http://blog.saymagic.cn/2015/02/16/release-library-to-jcenter.html 最外面的build.gradle // Top-leve ...

  8. onWindowFocusChanged

    这个onWindowFocusChanged指的是这个Activity得到或者失去焦点的时候 就会call. 也就是说 如果你想要做一个Activity一加载完毕,就触发什么的话 完全可以用这个!!! ...

  9. JAVA基础学习day18--常用工具类

    一.System 1.1.概述 System 类包含一些有用的类字段和方法.它不能被实例化. 在 System 类提供的设施中,有标准输入.标准输出和错误输出流:对外部定义的属性和环境变量的访问:加载 ...

  10. 【重构】 利用 cos 组件实现jsp中上传附件

    利用JSP&Servlet重构项目 利用 cos 组件实现jsp中上传附件 fileUpload.jsp --> FileUploadController.java --> fil ...