该文章会随着strust2 学习的深入,不断添加和更新,说是总结,不过很不全面,都是自己在学习中遇到的一些难点,或者比较难记住,应用时该注意的地方。。当然我是在初学者的角度上写的,深度很有限。。
有些需具体分析的总结,已经写在blog 的struts2 标签的文章中,下面的只是一些小的知识点。
ps:给出的网址仅仅是供深入理解参考用,基本需掌握的都已写在本文中。
--------------------------------------------------------------------------------
1、struts2 中用于处理用户请求的Action实例,并不是用户实现的业务控制器,而是Action代理。
在sturts2 的处理流程是: 页面请求->ServletDispatcher->创建Action代理类->通过一系列的拦截器处理->调用实际的业务处理控制器(Action)->result页面
从处理流程可看出,用户实现的Action类仅仅是struts2 的Action 代理的代理目标。用户实现的业务控制器(Action)则包含了对用户请求的处理。用户的求求数据包含在HttpServletRequest 对象里,而用户的Action 类无需反问HttpServletRequest对象,拦截器负责将HttpServletRequest 里的请求数据解析出来(放在AcitonContext中),并传给业务逻辑组件Actio实例。
有关数据的传输请参考:http://zhxing.iteye.com/blog/375823
参考的网址:http://www.iteye.com/wiki/struts2/1379-action-in-struts2
2、struts2 中 bean 的配置及作用
先来看下sitemesh 插件中 Struts-plugin.xml文件中bean配置是这样的:
<struts>
<bean
class="org.apache.struts2.sitemesh.FreeMarkerPageFilter"
static="true" optional="true"/>
<bean class="org.apache.struts2.sitemesh.VelocityPageFilter"
static="true" optional="true"/>
</struts>
Struts2框架以可配置的方式来管理Struts2的核心组件,从而允许开发者可以很方便地扩展该框架的核心组件。当开发者需要扩展,或者替换Struts2的核心组件时,只需要提供自己的组件类,并将该组件实现类部署在Struts2的IoC容器即可。
就是说框架在启动的时候会将一个Struts的FreemarkerManager类的实例注入到FreeMarkerPageFilter中去,那么这个FreemarkerManager类的实例从哪儿来呢?这正是我们在struts-default.xml定义的一个bean,如下所示:
<bean class="org.apache.struts2.views.freemarker.FreemarkerManager" name="struts" optional="true"/>
在struts.xml文件中定义Bean时,通常有如下2个作用:
* 创建Bean的实例,将该实例作为Struts2框架的核心组件使用
* Bean包含的静态方法需要一个值注入。
在第一中用法下,因为Bean实例往往是作为一个核心组件使用的,一次需要告诉Struts2容器该实例的作用――就是该实例实现了哪个接口,这个接口往往定义了该组件所必须遵守的规则。如:
<bean type="com.opensymphony.xwork2.ObjectFactory" name="myfactory" class="com.company.myapp.MyObjectFactory"/>
对于第二种用法,则可以很方便的允许不创建某个类的实例,缺可以接收框架常量。用法如下,通常需要设置static=”true”:
<bean class="org.apache.struts2.dispatcher.FilterDispatcher" static="true"/>
提示:对于绝大部分Struts2应用而言,我们无需要重新定义Struts2框架的核心组件,也就无需在文件中定义Bean。
Bean元素的一下几个属性:
* class:必填,实现类。
* type:可选,它指定了Bean实例实现的Struts2的规范,该规范通常是通过某个接口来体现的,一次该属性的值通常是一个Struts2接口。如果需要将Bean实例作为Struts2组件来使用,则应该指定该属性值。
* name:可选。
* scope:Bean实例的作用域。Default、 singleton 、request 、session和 thread 其中之一。
* static:是否使用静态方式注入。当指定type属性时,该属性不应该指定为true。
* optional:是否是一个可选Bean。可选属性。
关于bean 作用的网址:http://tech.it168.com/jd/2008-07-04/200807041113757.shtml
关于bean 配置的网址:http://aumy2008.blogbus.com/logs/14431770.html
3、Struts2 临时指定的国际化文件
关于在验证方面和其他的一些中涉及的国际化在很多文章中都能找到,这里就不写了,我只写在临时指定的国际化,弥补我学习中的遗漏。
通过临时指定资源文件的方式,可以在jsp页面中输出国际化消息时临时指定国际化资源的位置,在这种方式下需要借助struts2的另外一个标签:<s:i18n .../>
举个例子,一看该就明白了:该例子包括两份资源文件: tmp_zh_CN.properties 和 tmp_en_US.propertier ,两个文件的具体写法就不写了,用下面的文件内容转换下就行了,该两个文件保存在WEB-INF/classes 路径下:
文件的内容是:
loginPage=登陆页面 errorPage=错误页面 succPage=成功页面 failTip=全局消息:对不起,您不能登录! succTip=全局消息:欢迎,您已经登录! user=用户名 pass=密 码 login=登陆
JSP 文件的内容是:
<%@ page language="java" contentType="text/html; charset=GBK"%> <%@taglib prefix="s" uri="/struts-tags"%> <html> <head> <title> <!--使用i18n 作为 s:text 标签的父标签,临时指定国际化资源文件的baseName为 tmp--> <s:i18n name="tmp"> <!--输出国际化信息--> <s:text name="loginPage"/> </s:i18n> </title> </head> <body> <!--使用i18n 作为 s:text 标签的父标签,临时指定国际化资源文件的baseName为 tmp--> <s:i18n name="tmp"> <s:form action="Login"> <s:textfield name="username" key="user"/> <s:textfield name="password" key="pass"/> <s:submit key="login"/> </s:form> </s:i18n> </body> </html>
很简单,其他就不说了。。看了就明白了。
4、用户自行选择国际化文件语言
在struts2 中,我们可以通过ActionContext.getContext().setLocale(Locale arg) 设置用户的默认语言,不过,这种方式完全是一种手动方式,而且需要编程实现。
为了简化设置用户默认语言环境,struts2 提供了一个名为i18n 的拦截器,并且将其注册在默认的拦截器栈中(defaultStack),i18n 拦截器在执行Action 方法前,自动查找请求中一个名为request_locale 的参数,如果该参数存在,拦截器就将其作为参照,转换成locale 对象,并将其设为用户默认的locale。
除此以外,i18n 拦截器还会在将上面生成的locale 对象保存在用户session 的名字为"WW_TRANS_I18N_LOCALE" 的属性中,一旦用户session 中存在一个名为这个的属性,则该属性指定的locale将会作为浏览者的默认locale。
由此可看出,在页面中,只要更改session 中的"WW_TRANS_I18N_LOCALE" 的属性,即可改变浏览者的页面国际化语言。
实例:
jsp 页面(在页面中选择语言的select 涉及的国际化资源文件就不写了。):
<%@ page language="java" contentType="text/html; charset=GBK"%> <%@taglib prefix="s" uri="/struts-tags"%> <script type="text/javascript"> function langSelecter_onChanged() { document.getElementById("langForm").submit(); } </script> <!--将用户session 中的 WW_TRANS_I18N_LOCALE 属性设置成 SESSION_LOCALE --> <s:set name="SESSION_LOCALE" value="#session['WW_TRANS_I18N_LOCALE']"/> <!--使用lee.Locales ,得到locales 实例,并把它设在页面作用域内。 --> <s:bean id="locales" name="lee.Locales"> <s:param name="current" value="#SESSION_LOCALE == null ? locale : #SESSION_LOCALE"/> </s:bean> <form action="<s:url/>" id="langForm" style="background-color:#bbbbbb; padding-top: 4px; padding-bottom: 4px;"> <!--输出国际化提示 --> <s:text name="languag"/> <!--选择国际化语言 --> <s:select label="Language" list="#locales.locales" listKey="value" listValue="key" value="#SESSION_LOCALE == null ? locale : #SESSION_LOCALE" name="request_locale" id="langSelecter" onchange="langSelecter_onChanged()" theme="simple"/> </form>
lee.Locales 文件
package lee; import java.util.*; /** * @author yeeku.H.lee [email protected] * @version 1.0 * <br>Copyright (C), 2005-2008, yeeku.H.Lee * <br>This program is protected by copyright laws. * <br>Program Name: * <br>Date: */ public class Locales { private Locale current; public void setCurrent(Locale cur) { this.current = cur; } public Map<String, Locale> getLocales() { Map<String, Locale> locales = new Hashtable<String, Locale>(); ResourceBundle bundle = ResourceBundle.getBundle("messageResource" , current); locales.put(bundle.getString("usen"), Locale.US); locales.put(bundle.getString("zhcn"), Locale.CHINA); return locales; } }