本篇的主要内容是讲解struts.xml文件中每个元素的含义.
1) Bean配置
Struts2框架是一个高度可扩展的框架,框架的大部分核心组件,Struts2并不是直接以硬编码的方式写在代码中的,而是以自己的IOC(控制反转容器)来管理框架的核心组件.
Struts2框架以可配置的方式来管理Struts2的核心组件,从而允许开发者可以非常方便的扩展该框架的核心组件.当开发者需要扩展,或者替换Struts2的核心组件的时候,只需要提供自己的组件实现类,并且将组件实现类部署在Struts2的IOC容器当中就可以了.
打开Struts-core-2.1.6.jar压缩文件中的struts-default.xml文件,看到在该文件中配置了大量的Bean定义,该配置文件的代码如下:
<!—下面配置了Struts2的三个类型检测器Bean --> <bean type="com.opensymphony.xwork2.conversion.ObjectTypeDeterminer" name="tiger" class="com.opensymphony.xwork2.conversion.impl.DefaultObjectTypeDeterminer"/> <bean type="com.opensymphony.xwork2.conversion.ObjectTypeDeterminer" name="notiger" class="com.opensymphony.xwork2.conversion.impl.DefaultObjectTypeDeterminer"/> <bean type="com.opensymphony.xwork2.conversion.ObjectTypeDeterminer" name="struts" class="com.opensymphony.xwork2.conversion.impl.DefaultObjectTypeDeterminer"/> <!—下面配置了Struts2的两个文件上传器Bean --> <bean type="org.apache.struts2.dispatcher.multipart.MultiPartRequest" name="struts" class="org.apache.struts2.dispatcher.multipart.JakartaMultiPartRequest" scope="default"/> <bean type="org.apache.struts2.dispatcher.multipart.MultiPartRequest" name="jakarta" class="org.apache.struts2.dispatcher.multipart.JakartaMultiPartRequest" scope="default" /> <!—下面配置了Struts2的标签库处理的Bean --> <bean type="org.apache.struts2.views.TagLibrary" name="s" class="org.apache.struts2.views.DefaultTagLibrary" />
上面的配置文件配置了Struts2框架的内置Bean,其中有三个类型检测Bean,就是在前面struts.properties文件中允许使用的类型检测器.( struts.objectTypeDeterminer:该属性指定Struts2的类型检测机制,通常支持tiger和notiger两个属性值.).
在struts.xml文件中定义Bean的时候,通常有如下两个作用:
l 创建该Bean的实例,将该实例作为Struts2框架的核心组件使用.
l Bean包含的静态方法需要一个值的注入
在第一种用法下,因为Bean实例往往是作为一个核心组件使用的,因此需要告诉Struts2容器该实例的作用----就是该实例实现了哪个接口,这个接口往往定义了该组件所必须遵守的一个规范.
比如说下面的配置片段,我们使用了一个自定义的ObjectFactory来替代了Struts2内置的ObjectFactory.替换ObjectFactory的代码片段如下:
<?xml version="1.0" encoding="UTF-8"?> <!DOCTYPE struts PUBLIC "-//Apache Software Foundation//DTD Struts Configuration 2.1//EN" "http://struts.apache.org/dtds/struts-2.1.dtd"> <struts> <!-- 配置定制的ObjectFactory Bean,该Bean实现了 ObjectFactory接口,实现类是MyObjectFactory --> <bean type="com.opensymphony.xwork2.ObjectFactory" name="myfactory" class="com.supermos.app.MyObjectFactory"/> </struts>
对于第二种用法,可以很方便的不创建某个类的实例,却可以接受框架的常量,在这种方式下,通常要设置static=”true”.
下面是struts-default.xml配置文件的代码片段:
<struts> <bean class=”org.apache.struts.dispatcher.FilterDispatcher” static=”true”/> </struts>
(?为什么这个地方我没有在struts-default.xml(2.1.6版)中没有找到?莫非是被删除咧?)
对于大部分Struts2应用而言,我们不需要重新定义Struts2框架的核心组件,也就不需要在struts.xml文件中定义Bean.
正如前面看到的,使用<bean/>元素在struts.xml文件中定义Bean,Bean元素有如下几个属性:
l class:这个属性石必填属性,他指定了Bean实例的实现类
l type:这个属性是可选属性,它指定了Bean实例实现的Struts2的规范,该规范通常是通过某个接口来实现的,因此该属性的值通常就是一个Struts接口.如果需要将Bean实例作为Struts2的组件来使用的话,则应该制定该属性值.
l name:该属性指定了Bean实例的名字,对于有相同type类型的多个Bean则他们的name属性不能够相同,这个属性也是一个可选属性.
l scope:该属性指定Bean实例的作用域,该属性是一个可选的属性,属性值只能是default.singlton,request,session或者thread其中的一个.
l static:该属性指定Bean是否使用静态方法注入.通常而言,当指定了type属性的时候,该属性不应该指定为true.
l optional:该属性指定该Bean是否是一个可选的Bean,该属性是一个可选的属性.
2) 常量配置
在struts.xml文件中配置常量是一种指定Struts2属性的方式.配置Struts2属性有两种方式,一种是在struts.xml中配置,一种是在struts.properties中进行属性的配置.实际上,这两种方式的作用是基本相似的.
(通常都是在struts.xml文件当中来对Struts2进行属性的配置,一般不需要在struts.properties文件中定义Struts2属性,之所以要保留使用struts.properties文件定义Struts2属性的方式,主要是为了保持与WebWork的向后的兼容性)
这样看起来,Struts2的常量既可以在struts.xml文件中配置,也可以再struts.properties文件中配置.实际上,还可以在web.xml文件中配置Struts2常量.
通常,Struts2框架按照如下搜索顺序加载Struts2常量:
l struts-default.xml该文件保存在struts2-core-2.x..x.jar文件当中.
l struts-plugin.xml该文件保存在struts2-Xxx-2.x.x.jar等Struts2插件JAR文件当中
l struts.xml该文件是WEB应用默认的Struts2配置文件
l struts.properties该文件是web应用默认的Struts2配置文件
l web.xml该文件时web应用的配置文件
上面指定了Struts2框架搜索Struts2常量的顺序,如果在多个文件当中配置了同一个Struts2常量,则后一个文件中配置的常量就会覆盖前面文件中配置的常量.
在不同的文件中,配置常量的方式是不一样的,但是不管在哪个文件当中,配置Struts2常量都需要制定两个属性,常量name和常量value.
其中在Struts.xml文件中通过constant元素来配置常量,配置常量需要指定两个必填的属性:
① name:该属性指定了常量name
② value:该属性指定了常量value
例如,如果需要指定Struts2国际化资源文件的baseName为mess,那么就可以在Struts.xml文件中通过如下代码来进行配置:
<?xml version="1.0" encoding="UTF-8"?> <!DOCTYPE struts PUBLIC "-//Apache Software Foundation//DTD Struts Configuration 2.1//EN" "http://struts.apache.org/dtds/struts-2.1.dtd"> <struts> <constant name="struts.custom.i18n.resources" value="mess"/> </struts>
对于struts.properties文件而言,该文件的内容就是系列的key-value对,其中每个key对应一个Struts2的常量的name,每个value对应一个Struts2常量value.
比如说如下的代码段:
#指定了Struts2应用处于开发阶段
struts.devMode=true
上面的配置片段配置了名字为struts.devMode的Struts2常量,该常量值为true
在web.xml文件中配置Struts2常量,可以通过<filter>元素的<init-param>元素来指定,每个<init-param>元素配置了一个Struts2常量.
下面的代码在web.xml文件中,配置了Struts2应用所需要的国际化资源文件的baseName:
<?xml version="1.0" encoding="UTF-8"?> <web-app version="2.4" xmlns="http://java.sun.com/xml/ns/j2ee" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://java.sun.com/xml/ns/j2ee http://java.sun.com/xml/ns/j2ee/web-app_2_4.xsd"> <filter> <filter-name>Struts2</filter-name> <filter-class>org.apache.struts2.dispatcher.FilterDispatcher</filter-class> <init-param> <param-name>mess</param-name> <param-value>struts.custom.i18n.resources</param-value> </init-param> </filter> <filter-mapping> <filter-name>Struts2</filter-name> <url-pattern>/*</url-pattern> </filter-mapping> <welcome-file-list> <welcome-file>login.jsp</welcome-file> </welcome-file-list> </web-app>
在上面的web.xml文件当中,当配置FilterDispatcher的时候,还通过init-param子元素来配置了Struts2常量,指定了国际化资源文件的baseName为mess.
在实际开发中,Struts2的常量通常不配置在web.xml当中,如果采用这种配置在web.xml的形式,需要更多的代码,而且降低了web.xml文件的可读性,通常推荐将Struts2常量集中在struts.xml文件中进行集中的管理.
3) 包配置
Struts2框架中,核心组件就是Action,拦截器等等,Struts2框架使用包来管理Action和拦截器等等,每个包就是多个Action,多个拦截器,多个拦截器应用的集合
配置包的时候,必须要指定name属性,这个属性是引用这个包的key.除此之外,还可以指定一个可选的extends属性,extends属性值必须是另外一个包的属性.指定extends属性表示让该表继承其他包,子包可以继承一个或者多个父包中的拦截器,拦截器栈,action等配置.
除此之外,Struts2还提供了一种所谓的抽象包,抽象包的含义是该包不能包含Action定义.为了显示的指定一个包是抽象包,可以为该package元素增加abstract=”true”属性.
在struts.xml当中,package用于定义包配置,每个package元素定义了一个包配置,在定义package元素的时候可以指定如下几个属性:
l name:这是一个必填的属性,该属性指定了该包的名字,该名字是该包被其他包引用的key.
l extends:该属性是一个可选的属性,该属性指定该包继承其他的包.可以继承其他包中的Action定义,拦截器定义等等.
l namespace:该属性是一个可选的属性,该属性定义该包的命名空间.
l abstract:该属性是一个可选的属性,它指定该包是否是一个抽象包,抽象包不能包含Action定义.(因为Struts2的配置文件是从上到下处理的,所以父包应该放在子包前面来定义.)
下面是一个简单的struts.xml配置文件的范例,在下面的struts.xml文件当中配置了两个包,其中名为default的包,继承了Struts2框架的默认包struts-default.
<?xml version="1.0" encoding="UTF-8"?> <!DOCTYPE struts PUBLIC "-//Apache Software Foundation//DTD Struts Configuration 2.1//EN" "http://struts.apache.org/dtds/struts-2.1.dtd"> <struts> <!-- 配置第一个包,该包的名字为default继承自struts-default --> <package name="default" extends="struts-default"> <!-- 下面定义了拦截器部分 --> <interceptors> <!-- 定义拦截器栈 --> <interceptor-stack name="crudStack"> <interceptor-ref name="params"/> <interceptor-ref name="defaultStack"/> </interceptor-stack> </interceptors> <default-action-ref name="showcase"/> <!-- 定义一个Action,该Action直接映射到showcase.jsp页面 --> <action name="Showcase"> <result>showcase.jsp</result> </action> <!-- 定义了一个Action,该Action类为com.opensymphony.webwork.showcase.DateAction --> <action name="Date" class="supermos.DateAction"> <result name="success">/date.jsp</result> </action> </package> <!-- 定义名字为skill的包,该Action继承自default的包 --> <package name="skill" extends="default" namespace="/skill"> <!-- 定义默认的拦截器引用 --> <default-interceptor-ref name="crudStack"/> <!-- 定义名为Edit的Action,该action对应的处理类为superSkillAction --> <action name="Edit" class="super.SkillAction"> <result>/empmanager/editSkill.jsp</result> <interceptor-ref name="params"/> <interceptor-ref name="basicStack"/> </action> <!-- 定义名为Save的Action,该action对应的处理类为super.SkillAction,使用save方法作为处理的方法 --> <action name="Save" class="super.SkillAction" method="save"> <result name="input">/empmanager/editSkill.jsp</result> <result name="redirect">edit.action?skillName=${currentSkill.name}</result> </action> <!-- 定义名为Delete的Action,该Action对应的处理类为supermos.SkillAction,使用delete方法作为处理方法 --> <action name="Delete" class="supermos.SkillAction"> <result name="error">/empmanager/editSkill.jsp</result> <result type="redirect">edit.action?SkillName=${currentSkill.name}</result> </action> </package> </struts>
在上面的配置文件当中,配置了两个包,其中定义名为skill的包的时候,还制定了改包的命名空间skill,其中skill用于指定某个包对应的命名空间.
4) 命名空间的配置
考虑到在同一个Web应用中,需要同名的Action,Struts2以命名空间的方式来管理Action,在同一个命名空间中不能有同名的Action,不同的命名空间中可以存在同名的Action.
Struts2不支持为单独的Action设置命名空间,而是通过为包指定namespace属性来为包下面的所有的Action指定共同的命名空间.
看下面struts.xml配置文件的源代码:
//TODO
在上面的struts.xml配置文件当中,配置了两个包,supermos和get包,配置get包的时候指定了该包的命名空间为/book.
对于名字为supermos的包而言,没有指定namespace属性,如果某个包没有指定namespace属性,既该包使用了默认的命名空间,则默认的命名空间总是””.
当某个包指定了命名空间之后,该包下的所有的Action处理的URL应该是命名空间+Action名,以上名为get的包为例,该包下包含了名为getBooks的Action,则该Action处理的URL为:
http://localhost:8080/namespace/book/GetBooks.action
从上面可以看出,Struts2命名空间的作用类似于Struts1里面模块的作用.
(Struts2的命名空间的作用就类似于Struts1里模块的作用,它允许以模块化的方式来组织Action)
除此之外,Struts2还可以显示指定根命名空间,通过设置某个包的namespace=”/”来制定跟命名空间.
如果请求为/barspace/bar.action,系统会首先去查找/barspace命名空间里面的名字为bar的Action,如果在该命名空间里面找到对应的Action,则使用该Action处理该请求,否则,系统将到默认命名空间里面找到名字为bar的action,如果找到对应的Action,则使用该Action处理用户请求,如果两个命名空间里面都找不到名字为bar的Action,则系统出现错误.
(默认空间里面的Action可以处理任何模块下的Action请求,意思是说,如果存在URL为/barspace/bar.action的请求,并且/barspace的命名空间下没有名为bar的Action,则命名空间下名为bar的Action也会处理用户请求.)
如果请求为/login.action,系统会在根命名空间下(“/”)查找名字为login的Action,如果在根命名空间下找到了名字为login的Action,则由该Action处理用户的请求,否则系统将转入默认空间中查找名字为login的Action,如果默认的命名空间中有名字为login的Action,则由该Action处理用户请求,如果两个命名空间里面都找不到名字为login的Action,那么系统将会出现错误.
(命名空间只有一个级别,如果请求的URL是/bookservice/search/get.action,系统将会首先在/bookservice/search的命名空间下查找名字为get的Action,如果在该命名空间中找到名字为get的Action,则由该Action处理用户的请求,如果在该命名空间中没有找到get的Action,系统将会直接进入默认的命名空间中查找名字为get的Action,而不会在/bookService的命名空间下查找名字为get的Action).
5) 包含配置
包含配置体现的是软件工程中的”分而治之”的原则.
Struts2允许将一个配置文件分解成多个配置文件,从而提供配置文件的可读性.一旦我们通过多个struts.xml文件来配置Action,但是Struts2默认只加载WEB-INF/classes下面的struts.xml,我们就必须通过struts.xml文件来包含其他的配置文件.
在struts.xml文件中包含其他配置文件通过<include…/>元素来完成,配置<include…/>元素需要指定一个必须的file属性,该属性指定了被包含配置文件的文件名,西面是一个struts.xml文件中包含了4个配置文件的代码片段:
<?xml version="1.0" encoding="UTF-8"?> <!DOCTYPE struts PUBLIC "-//Apache Software Foundation//DTD Struts Configuration 2.0//EN" "http://struts.apache.org/dtds/struts-2.0.dtd"> <struts> <include file="struts-user1.xml"/> <include file="struts-user2.xml"/> <include file="struts-user3.xml"/> <include file="struts-user4.xml"/> <include file="struts-user5.xml"/> </struts>
被包含的struts-user1.xml,struts-user2.xml等配置文件,是标准的Struts2的配置文件,一样都包含了DTD信息,Struts2配置文件的根元素等信息.通常,将Struts2的所有的配置文件都放在Web应用的WEB-INF/classes路径的下面,struts.xml文件包含了其他的配置文件,Struts2框架自动的加载struts.xml文件的时候,从而完成了加载所有的配置文件的信息.
6) 包拦截器配置
拦截器其实就是AOP(面向切面编程)的编程思想,拦截器允许在Action处理之前,或者Action处理结束之后,插入开发者自己定义的代码.
在很多时候,我们需要在多个Action进行相同的操作,比如说权限的控制,此处就可以使用拦截器来检查用户是否登录,用户的权限是否足够(当然,这个地方也可以借助spring的Aop框架来完成).通常,使用拦截器可以完成如下的几个操作:
l 进行权限的控制(检查浏览者是否登录,并且有足够的访问权限)
l 跟踪日志(记录每个浏览者所请求的每个Action)
l 跟踪系统的性能瓶颈(我们可以通过记录每个Action开始处理的时间和处理结束的时间,从而获得耗时比较长的Action)
Struts2也允许将多个拦截器组合在一起,从而形成一个拦截器栈.一个拦截器栈可以包含多个拦截器,多个拦截器组成一个拦截器栈,对于Struts2系统而言,多个拦截器组成的拦截器栈对外也表现成一个拦截器.
(可以认为多个拦截器组成的拦截器栈是一个比较大的拦截器)
定义拦截器栈之前,必须首先定义组成拦截器栈的多个拦截器,Struts2吧拦截器栈当成拦截器来处理,因此拦截器栈和拦截器都放在<interceptors…/>元素中进行定义.
下面是拦截器的定义片段:
<?xml version="1.0" encoding="UTF-8"?> <!DOCTYPE struts PUBLIC "-//Apache Software Foundation//DTD Struts Configuration 2.0//EN" "http://struts.apache.org/dtds/struts-2.0.dtd"> <struts> <package name="supermos" extends="struts-default"> <interceptors> <!-- 定义权限检查的拦截器 --> <interceptor name="authority" class="super.AuthrityInterceptor"/> <!-- 定义日志记录的拦截器 --> <interceptor name="log" class="super.LogInterceptor"/> <!-- 定义一个拦截器栈 --> <interceptor-stack name="authorityandlog"> <!-- 定义该拦截器里面包含的authority拦截器 --> <interceptor-ref name="authority"/> <!-- 定义该拦截器里面包含的log拦截器 --> <interceptor-ref name="log"/> </interceptor-stack> </interceptors> </package> </struts> 在上面的拦截器配置片段中,定义了两个拦截器,并且将两个拦截器组成一个拦截器栈,一旦定义了拦截器和拦截器栈以后,在Action中使用拦截器或者拦截器栈的方式是相同的. <?xml version="1.0" encoding="UTF-8"?> <!DOCTYPE struts PUBLIC "-//Apache Software Foundation//DTD Struts Configuration 2.0//EN" "http://struts.apache.org/dtds/struts-2.0.dtd"> <struts> <package name="supermos" extends="struts-default"> <interceptors> <!-- 定义权限检查的拦截器 --> <interceptor name="authority" class="super.AuthrityInterceptor"/> <!-- 定义日志记录的拦截器 --> <interceptor name="log" class="super.LogInterceptor"/> <!-- 定义一个拦截器栈 --> <interceptor-stack name="authorityandlog"> <!-- 定义该拦截器里面包含的authority拦截器 --> <interceptor-ref name="authority"/> <!-- 定义该拦截器里面包含的log拦截器 --> <interceptor-ref name="log"/> </interceptor-stack> </interceptors> <action name="MyAction" class="supermos.MyAction"> <result name="success">/success.jsp</result> <interceptor-ref name="authorityandlog"/> </action> </package> </struts>
上面的配置文件中定义了一个名字为MyAction的Action,并且在该Action中引用了一个名字为authorityandlog的拦截器栈,对于Action而言,引用拦截器和引用拦截器栈的用法是完全一样的.