详解Webwork中Action 调用的方法
详解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 每次响应请求的入口方法:
- 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 参数,并对 request 参数进行封装,这次封装主要是针对多媒体请求进行的特殊处理,例如项目中的文件上传请求,导出各种类型文件等...
•包装完 request 之后,service 方法调用 ServletDispatche.serviceAction() 方法,并调用 getApplicationMap、getSessionMap、getRequestMap、 getParameterMap、getActionName、getNameSpace 6 个方法开始了Action 业务逻辑调用前的前戏。
•getNameSpace 方法用来获得一个Action所属的名称空间,例如 : "/my/MyAction.action"则返回"/my",具体实现如下:
- 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",具体实现如下:
- 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);
- }
- protected Map getRequestMap(HttpServletRequest request){
- return new RequestMap(request);
- }
•getParameterMap 方法返回一个包含请求中所有参数的Map,具体代码如下:
- protected Map getParameterMap(HttpServletRequest request) throws IOException{
- return request.getParameterMap();
- }
•getSessionMap 方法返回一个包含 session 中所有属性的 Map,具体实现类是 SessionMap,具体代码如下:
- protected Map getSessionMap(HttpServletRequest request){
- return new SessionMap(request);
- }
•getApplicationMap 方法返回一个包含 Application 中所有属性的Map,具体实现类 是ApplicationMap,具体代码如下:
- protected Map getApplicationMap(){
- return new ApplicationMap(getServletContext());
- }
•WebWork之所以要把request 的属性、参数,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 调用了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 实例,代码如下:
- static ActionProxyFactory factory = new DefaultActionProxyFactory();
- public static void setFactory(ActionProxyFactory factory){
- factory = factory;
- }
- public static ActionProxyFactory getFactory(){
- return factory;
- }
• DefaultActionProxyFactory 的 createActionProxy 方法返回了 DefaultActionProxy 实例。
- public ActionProxy createActionProxy(String namespace, String actionName, Map extraContext)throws Exception {
- setupConfigIfActionIsCommand(namespace, actionName);
- return new DefaultActionProxy(namespace, actionName, extraContext, true);
- } •DefaultActionProxy的构造函数
- 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 的名字等参数赋予本地变量,接着通过 ConfigurationManager 获得当前请求的 Action 的配置信息[这里在5中已经描述过]。接着调用自身的 prepare 方法创建一个 ActionInvocation 对象赋予自身变量 invocation。在之后的 execute 方法中通过操纵invocation 来实现我们自己写的Action 的调用。
- protected void prepare() throws Exception {
- this.invocation = ActionProxyFactory.getFactory().createActionInvocation(this, this.extraContext);
- }
详解Webwork中Action 调用的方法的更多相关文章
- [原创]java WEB学习笔记55:Struts2学习之路---详解struts2 中 Action,如何访问web 资源,解耦方式(使用 ActionContext,实现 XxxAware 接口),耦合方式(通过ServletActionContext,通过实现 ServletRequestAware, ServletContextAware 等接口的方式)
本博客的目的:①总结自己的学习过程,相当于学习笔记 ②将自己的经验分享给大家,相互学习,互相交流,不可商用 内容难免出现问题,欢迎指正,交流,探讨,可以留言,也可以通过以下方式联系. 本人互联网技术爱 ...
- 详解Java中的Object.getClass()方法
详解Object.getClass()方法,这个方法的返回值是Class类型,Class c = obj.getClass(); 通过对象c,我们可以获取该对象的所有成员方法,每个成员方法都是一个Me ...
- 详解JavaScript中的arc的方法
今天说说JavaScript在网页中画圆的函数arc! 一.arc所需要的参数设置 1 arc(x, y, radius, startAngle, endAngle, counterclockwise ...
- 详解 JS 中 new 调用函数原理
JavaScript 中经常使用构造函数创建对象(通过 new 操作符调用一个函数),那在使用 new 调用一个函数的时候到底发生了什么?先看几个例子,再解释背后发生了什么. 1)看三个例子 1.1 ...
- 详解Java中的clone方法
详解Java中的clone方法 参考:http://blog.csdn.net/zhangjg_blog/article/details/18369201/ 所谓的复制对象,首先要分配一个和源对象同样 ...
- 举例详解Python中的split()函数的使用方法
这篇文章主要介绍了举例详解Python中的split()函数的使用方法,split()函数的使用是Python学习当中的基础知识,通常用于将字符串切片并转换为列表,需要的朋友可以参考下 函数:sp ...
- jQuery:详解jQuery中的事件(二)
上一篇讲到jQuery中的事件,深入学习了加载DOM和事件绑定的相关知识,这篇主要深入讨论jQuery事件中的合成事件.事件冒泡和事件移除等内容. 接上篇jQuery:详解jQuery中的事件(一) ...
- 【转】详解C#中的反射
原帖链接点这里:详解C#中的反射 反射(Reflection) 2008年01月02日 星期三 11:21 两个现实中的例子: 1.B超:大家体检的时候大概都做过B超吧,B超可以透过肚皮探测到你内 ...
- MySQL关闭过程详解和安全关闭MySQL的方法
MySQL关闭过程详解和安全关闭MySQL的方法 www.hongkevip.com 时间: -- : 阅读: 整理: 红客VIP 分享到: 红客VIP(http://www.hongkevip.co ...
随机推荐
- 《JAVA与模式》之组合模式
定义(GoF<设计模式>): 将对象组合成树形结构以表示“部分整体”的层次结构.组合模式使得用户对单个对象和使用具有一致性. 及角色: 1.Component 是组合中的对象声明接口,在适 ...
- R语言-神经网络包RSNNS
code{white-space: pre;} pre:not([class]) { background-color: white; }if (window.hljs && docu ...
- linux下log4j乱码解决
使用log4j的时候,在WIN系统的时候正常显示中文,但是发布到linux系统的时候中文就显示成乱码了 由于log4j配置文件中没有设置编码格式(encoding),所以log4j就使用系统默认编码. ...
- 使用WebMatrix发布网站
使用WebMatrix发布网站 WebMatrix 简介: Microsoft WebMatrix 是微软最新的 Web 开发工具,它包含了构建网站所需要的一切元素.您可以从开源 Web 项目或者内置 ...
- Microsoft Dynamics CRM 前瑞开发
做CRM开发最大的感受就是其前瑞开发过程中,调试起来比较麻烦,需要做一些断点还要配制一些浏览器设置,对新手来说比较困难.还有就是对REST调试,经常为了调试一个正确的结果而花费大量的时间.现在推荐一个 ...
- [ html canvas globalCompositeOperation ] canvas绘图属性 设置合成图像如何显示 属性演示
<!DOCTYPE html> <html lang='zh-cn'> <head> <title>Insert you title</title ...
- 上传Android代码到Jcenter(解决了字符映射的问题)
请先阅读:http://blog.saymagic.cn/2015/02/16/release-library-to-jcenter.html 最外面的build.gradle // Top-leve ...
- onWindowFocusChanged
这个onWindowFocusChanged指的是这个Activity得到或者失去焦点的时候 就会call. 也就是说 如果你想要做一个Activity一加载完毕,就触发什么的话 完全可以用这个!!! ...
- JAVA基础学习day18--常用工具类
一.System 1.1.概述 System 类包含一些有用的类字段和方法.它不能被实例化. 在 System 类提供的设施中,有标准输入.标准输出和错误输出流:对外部定义的属性和环境变量的访问:加载 ...
- 【重构】 利用 cos 组件实现jsp中上传附件
利用JSP&Servlet重构项目 利用 cos 组件实现jsp中上传附件 fileUpload.jsp --> FileUploadController.java --> fil ...