Java之旅Struts系列(3)——Struts2(前奏篇)

2024-05-26 07:38

本文主要是介绍Java之旅Struts系列(3)——Struts2(前奏篇),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

  上篇博客我们已经讲到了Struts1的Action类这种侵入式设计使得代码的复用率极低,那么是不是有别的解决方案呢,答案当然是有的。

  可能有的同学已经想到了是Struts2,也就是我们本篇要讲的,但是在讲解这个之前,我们先来了解一个别的mvc模式的框架:WebWork,了解了这个之后我们再来了解Struts2,效果肯定会更加的明显和高效。

  WebWork来自于另一个优秀的开源组织,相对于Struts1中的严重耦合或者说依赖,WebWork显然要更加的优秀,去除了Action与Servlet API 的耦合,这样也便于我们进行测试。

  对比与Struts1,WebWork在表现层支持方法支持的技术比较丰富,此外面对于Struts1依赖于web容器进行初始化的现象,WebWork是可以脱离web应用进行工作的,它拥有自己的控制反转(Inversion of Control)容器,还有就是WebWork2使用OGNL这个强大的语言表达式,可以访问值栈,OGNL强有力支持集合和索引。

  与Struts1处理流程是相似的,同样存在核心控制器和业务逻辑控制器,但是这里的核心控制器不再是ActionServlet而是ServletDispatcher,这个由框架进行提供,而业务逻辑Action由程序员进行提供。

  下面进行了一下关于WebWork的讲解。

 1.      WebWork流程详解

  我们先来看一个流程图:


  具体是怎么工作的呢?流程详解:

  当用户向web应用发送一个http请求的时候,该请求经过ActionContextCleanUp、SiteMesh等过滤器,再由WebWork的核心控制器ServletDispatcher进行拦截,如果用户的请求需要WebWork的业务逻辑控制器处理,该控制器则调用Action映射器,该映射器将用户的请求转发到对应的业务逻辑控制器,但是我们可以根据图示发现,此时的业务逻辑控制器并不是我们开发实现的控制器,而是WebWork创建的代理控制器。

  创建的这些需要在WebWork的核心配置文件xwork.xml中进行相关的定义,使得控制器以用户实现的控制器作为目标,以拦截器链中拦截器作为处理。此时代理控制器由系统代为实现,无需我们的参加。

  因为此时开发者实现的是WebWork控制器的目标,这样就使得WebWork的action可以与Servlet API进行分离,当自己的Action处理完用户的请求后,返回一个普通的字符串名称,从而在配置文件中将此名称指向相应的视图资源。

  与Struts1相比,WebWork存在很多的优势之处,不用与ServletAPI耦合


  此时我们看到Action实例只是实现了一个Action接口,并且实现其中的方法。

  此时我们发现在execute方法中没有了Struts1中的继承来的四个参数,这样就去除了与Struts1API以及Servlet API的依赖,Action中的参数与httpServletRequest无关,我们可以看到该action封装了表单上的两个参数,这两个参数初始化由Action拦截器负责,以用户请求参数为其进行赋值。

  WebWork很好滴解除了依赖,我们可以看到即使我们访问了http session ,但是还是没有出现相应的关于session的api,而是通过map的形式通过值栈进行读取。

  总之,Action是不必要与WebWork进行耦合的,这样使得代码的重用率得到了提高。其实在WebWork 中的action更像是一个pojo对象,虽然实现了Action接口,但是也仅仅是包含了一个execute方法,可以对比与Struts1中Action,我们可以看到继承来的Action具有严重的依赖性不仅仅依赖于Struts1 API,而且依赖于ServletAPI,此外继承和实现接口相比,继承一个父类之后意味着不能再继承,但是实现接口后还是可以再继承呢,便于扩展。

 2.      标签库—OGNL

  与Struts1存在相似之处,WebWork也提供了丰富的标签库,在讲解这个之前我们最需要了解的是一个强大的语言支持OGNL.

  什么是OGNL?

  OGNL是Object-Graph Navigation Language的缩写,它是一种功能强大的表达式语言,通过它简单一致的表达式语法,可以存取对象的任意属性,调用对象的方法,遍历整个对象的结构图,实现字段类型转化等功能。它使用相同的表达式去存取对象的属性。这样可以更好的取得数据。

  webwork2和现在的Struts2.x中使用OGNL取代原来的EL来做界面数据绑定,所谓界面数据绑定,也就是把界面元素(例如一个textfield,hidden)和对象层某个类的某个属性绑定在一起,修改和显示自动同步。

  WebWork表现层采用JSP,用WebWork标签库来支持OGNL表达式,其中我们比较常用输出对象的符号包括三个:# $ %。那么使用场景是怎么样的呢?

  “#”用途:

  访问的对象是OGNL上下文和Action上下文,#相当于ActionContext.getContext();是一种以map形式进行取值的用法。

比如在action实例中进行定义的时候

<span style="font-family:Microsoft YaHei;font-size:14px;"><strong>/* 列表显示 */publicString list() throwsException {List<Role>roleList = roleService.findAll();// 将获取到的列表值放到值栈里面ActionContext.getContext().put("roleList",roleList);return "list";}</strong></span>

  此时我们将值放在了map中,在jsp页面用表达式进行获取的时候是(以Struts2进行讲解):

  <s:iteratorvalue="#roleList">,此时获取到map中的值时我们用的是#;

  OGNL上下文实际上就是一个Map对象,由ognl.OgnlContext类表示。它里面可以存放很多个JavaBean对象。它有一个上下文根对象。

  上下文中的根对象可以直接使用名来访问或直接使用它的属性名访问它的属性值。否则要加前缀“#key”。可能有人说这个不是一个根对象吗?为什么是用#roleList。

  分下面的情况,值得注意的是:

   1.如果需要引用valueStack中的值,需要使用这样的形式。

   <s:iteratorvalue="# roleList " /> //userList在action部分被保存在Request中,所以使用#加属性名来引用值。

   2.如果集合的值是通过action的方法,假设我们的action中有一个getListMenu方法,返回一个List集合。

   我们可以使用如下的形式来引用这个集合,并用s:iterator来输出。

   <s:iteratorvalue="listMenu" />

  “$“用途

  在Struts 2配置文件中,引用OGNL表达式,经常会使用这个符号,比如页面的跳转需要添加完毕再次返回上一级页面,此时我们就需要配置相应的父ID了。

       <!--部门管理--><action name="department_*" class="departmentAction" method="{1}"><result name="list">/WEB-INF/jsp/departmentAction/list.jsp</result><result name="saveUI">/WEB-INF/jsp/departmentAction/saveUI.jsp</result><result name="toList" type="redirectAction">department_list?parentId=${parentId}</result></action>

  department_list?parentId=${parentId}用户返回到上一级的页面。

  “%”用途

  %符号的用途是在标志的属性为字符串类型时,计算OGNL表达式的值。

<s:form action="role_%{ id == null? 'add' : 'edit' }">

  这些是我们在项目中一些应用,具体的还有待继续学习,不过在此推荐比较好的文章。

  http://blog.csdn.net/tjcyjd/article/details/6850203

  与el表达式的区别

  EL:Expression Language 表达式语言,最早使用于JSP,灵感来自于 ECMAScript 和 XPath 表达式语言,它提供了在 JSP 中简化表达式的方法(jsp2.0)

  OGNL:Object-Graph Navigation Language,是一种功能强大的表达式语言,在struts2和webwork中使用是OGNL。具体用法的区别大家可以去查找相应的资料。网上资料很详细,我这里不再进行赘述了。

  了解了上面的这些,进入我们Struts2的学习,就比较轻松了。

这篇关于Java之旅Struts系列(3)——Struts2(前奏篇)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1003846

相关文章

SpringBoot全局域名替换的实现

《SpringBoot全局域名替换的实现》本文主要介绍了SpringBoot全局域名替换的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一... 目录 项目结构⚙️ 配置文件application.yml️ 配置类AppProperties.Ja

Java使用Javassist动态生成HelloWorld类

《Java使用Javassist动态生成HelloWorld类》Javassist是一个非常强大的字节码操作和定义库,它允许开发者在运行时创建新的类或者修改现有的类,本文将简单介绍如何使用Javass... 目录1. Javassist简介2. 环境准备3. 动态生成HelloWorld类3.1 创建CtC

JavaScript中的高级调试方法全攻略指南

《JavaScript中的高级调试方法全攻略指南》什么是高级JavaScript调试技巧,它比console.log有何优势,如何使用断点调试定位问题,通过本文,我们将深入解答这些问题,带您从理论到实... 目录观点与案例结合观点1观点2观点3观点4观点5高级调试技巧详解实战案例断点调试:定位变量错误性能分

Java实现将HTML文件与字符串转换为图片

《Java实现将HTML文件与字符串转换为图片》在Java开发中,我们经常会遇到将HTML内容转换为图片的需求,本文小编就来和大家详细讲讲如何使用FreeSpire.DocforJava库来实现这一功... 目录前言核心实现:html 转图片完整代码场景 1:转换本地 HTML 文件为图片场景 2:转换 H

Java使用jar命令配置服务器端口的完整指南

《Java使用jar命令配置服务器端口的完整指南》本文将详细介绍如何使用java-jar命令启动应用,并重点讲解如何配置服务器端口,同时提供一个实用的Web工具来简化这一过程,希望对大家有所帮助... 目录1. Java Jar文件简介1.1 什么是Jar文件1.2 创建可执行Jar文件2. 使用java

SpringBoot实现不同接口指定上传文件大小的具体步骤

《SpringBoot实现不同接口指定上传文件大小的具体步骤》:本文主要介绍在SpringBoot中通过自定义注解、AOP拦截和配置文件实现不同接口上传文件大小限制的方法,强调需设置全局阈值远大于... 目录一  springboot实现不同接口指定文件大小1.1 思路说明1.2 工程启动说明二 具体实施2

Java实现在Word文档中添加文本水印和图片水印的操作指南

《Java实现在Word文档中添加文本水印和图片水印的操作指南》在当今数字时代,文档的自动化处理与安全防护变得尤为重要,无论是为了保护版权、推广品牌,还是为了在文档中加入特定的标识,为Word文档添加... 目录引言Spire.Doc for Java:高效Word文档处理的利器代码实战:使用Java为Wo

SpringBoot日志级别与日志分组详解

《SpringBoot日志级别与日志分组详解》文章介绍了日志级别(ALL至OFF)及其作用,说明SpringBoot默认日志级别为INFO,可通过application.properties调整全局或... 目录日志级别1、级别内容2、调整日志级别调整默认日志级别调整指定类的日志级别项目开发过程中,利用日志

Java中的抽象类与abstract 关键字使用详解

《Java中的抽象类与abstract关键字使用详解》:本文主要介绍Java中的抽象类与abstract关键字使用详解,本文通过实例代码给大家介绍的非常详细,感兴趣的朋友跟随小编一起看看吧... 目录一、抽象类的概念二、使用 abstract2.1 修饰类 => 抽象类2.2 修饰方法 => 抽象方法,没有

SpringBoot 多环境开发实战(从配置、管理与控制)

《SpringBoot多环境开发实战(从配置、管理与控制)》本文详解SpringBoot多环境配置,涵盖单文件YAML、多文件模式、MavenProfile分组及激活策略,通过优先级控制灵活切换环境... 目录一、多环境开发基础(单文件 YAML 版)(一)配置原理与优势(二)实操示例二、多环境开发多文件版