您的位置:首页 > 编程语言 > Java开发

struts2中struts.xml配置文件详解

2017-03-16 12:39 337 查看
                               struts.xml的常用配置
              
                
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE struts PUBLIC "-//Apache Software Foundation//DTD Struts Configuration 2.3//EN" "http://struts.apache.org/dtds/struts-2.3.dtd">
<struts>
<!-- 所有匹配*.action的请求都由struts2处理 -->
<constant name="struts.action.extension" value="action" />
<!-- 是否启用开发模式 -->
<constant name="struts.devMode" value="true" />
<!-- struts配置文件改动后,是否重新加载 -->
<constant name="struts.configuration.xml.reload" value="true" />
<!-- 设置浏览器是否缓存静态内容 -->
<constant name="struts.serve.static.browserCache" value="false" />
<!-- 请求参数的编码方式 -->
<constant name="struts.i18n.encoding" value="utf-8" />
<!-- 每次HTTP请求系统都重新加载资源文件,有助于开发 -->
<constant name="struts.i18n.reload" value="true" />
<!-- 文件上传最大值 -->
<constant name="struts.multipart.maxSize" value="104857600" />
<!-- 让struts2支持动态方法调用 -->
<constant name="struts.enable.DynamicMethodInvocation" value="true" />
<!-- Action名称中是否还是用斜线 -->
<constant name="struts.enable.SlashesInActionNames" value="false" />
<!-- 允许标签中使用表达式语法 -->
<constant name="struts.tag.altSyntax" value="true" />
<!-- 对于WebLogic,Orion,OC4J此属性应该设置成true -->
<constant name="struts.dispatcher.parametersWorkaround" value="false" />

<package name="basePackage" extends="struts-default">

</package>

</struts>


  

<?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节点是struts2中组件化的方式 可以将每个功能模块独立到一个xml配置文件中 然后用include节点引用 -->
<include file="struts-default.xml"></include>

<!-- package提供了将多个Action组织为一个模块的方式
package的名字必须是唯一的 package可以扩展 当一个package扩展自
另一个package时该package会在本身配置的基础上加入扩展的package
的配置 父package必须在子package前配置
name:package名称
extends:继承的父package名称
abstract:设置package的属性为抽象的 抽象的package不能定义action 值true:false
namespace:定义package命名空间 该命名空间影响到url的地址,例如此命名空间为/test那么访问是的地址为http://localhost:8080/struts2/test/XX.action
-->
<package name="com.kay.struts2" extends="struts-default" namespace="/test">
<interceptors>
<!-- 定义拦截器
name:拦截器名称
class:拦截器类路径
-->
<interceptor name="timer" class="com.kay.timer"></interceptor>
<interceptor name="logger" class="com.kay.logger"></interceptor>
<!-- 定义拦截器栈 -->
<interceptor-stack name="mystack">
<interceptor-ref name="timer"></interceptor-ref>
<interceptor-ref name="logger"></interceptor-ref>
</interceptor-stack>
</interceptors>

<!-- 定义默认的拦截器 每个Action都会自动引用
如果Action中引用了其它的拦截器 默认的拦截器将无效 -->
<default-interceptor-ref name="mystack"></default-interceptor-ref>

<!-- 全局results配置 -->
<global-results>
<result name="input">/error.jsp</result>
</global-results>

<!-- Action配置 一个Action可以被多次映射(只要action配置中的name不同)
name:action名称
class: 对应的类的路径
method: 调用Action中的方法名
-->
<action name="hello" class="com.kay.struts2.Action.LoginAction">
<!-- 引用拦截器
name:拦截器名称或拦截器栈名称
-->
<interceptor-ref name="timer"></interceptor-ref>

<!-- 节点配置
name : result名称 和Action中返回的值相同
type : result类型 不写则选用superpackage的type struts-default.xml中的默认为dispatcher
-->
<result name="success" type="dispatcher">/talk.jsp</result>
<!-- 参数设置
name:对应Action中的get/set方法
-->
<param name="url">http://www.sina.com</param>
</action>
</package>
</struts>


 

一个Action内包含多个请求处理方法的处理

Struts1提供了DispatchAction,从而允许一个Action内包含多个请求处理方法。Struts2也提供了类似的功能。

处理方式主要有以下三种方式:

 

[b]1. 1   动态方法调用:[/b]

  DMI:Dynamic Method Invocation 动态方法调用。
  动态方法调用是指:表单元素的action不直接等于某个Action的名字,而是以感叹号后加方法名来指定对应的动作名:

    <!-- 动态方法调用HTML标签与Struts2标签 -->
<form action="computeAction!add.action" name="from" >
<s:form action="computeAction!add.action" name="form" theme="simple" >


  则用户的请求将提交到名为”computeAction”的Action实例,Action实例将调用名为”add”方法来处理请求。

  当指定调用某一方法来处理请求时,就不会走默认执行处理请求的execute()方法。

  注意:要使用动态方法调用,必须设置Struts2允许动态方法调用,通过设置struts.enable.DynamicMethodInvocation常量来完成,该常量属性的默认值是true。

<struts>
<!--
//禁用动态方法调用,默认为true启用,false禁用
constant:name="struts.enable.DynamicMethodInvocation"
-->
<constant name="struts.enable.DynamicMethodInvocation" value="true" />
</struts>


 

示列:简单的一个加法和减法例子。

1. index.jsp用户在页面输入两个数字,选择相加,或者相减

   当用户点击加或减需要走同一个Action但处理请求方法不同,这里使用了js动态选择。

<body>
<!-- 动态方法调用    使用:Struts2标签也可以使用HTML标签 -->
<s: name="form" theme="simple" >

num1:<s:textfield name="num1" />
num2:<s:textfield name="num2" />
<s:submit type="button" value="加" onclick="computeMethod('add')" />
<s:submit type="button" value="减" onclick="computeMethod('subtract')" />
</s:form>

<!-- js -->
<script type="text/javascript">
function computeMethod(op){
document.form.action="computeAction!"+op;//动态选择处理请求的方法
document.form.submit();//提交
}
</script>

</body>


 

2. struts.xml配置信息,启用动态方法调用(可选)

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE struts PUBLIC "-//Apache Software Foundation//DTD Struts Configuration 2.1//EN" "struts-2.1.dtd" >
<struts>
<!--
//禁用动态方法调用,默认为true启用,false禁用
constant:name="struts.enable.DynamicMethodInvocation"
-->
<constant name="struts.enable.DynamicMethodInvocation" value="true" />
<package name="struts2" extends="struts-default">
<action name="computeAction" class="com.struts.ComputeAction" >
<result name="fruitPage" >/fruit.jsp</result>
</action>
</package>
</struts>


 

3. ComputeAction控制器的类处理请求

package com.struts;
/**
* Struts2控制器的类
* @author asus
*
*/
public class ComputeAction {

/** 属性 */
private int num1;
private int num2;
private int fruit;//结果

/** 若请求为指定操作方法默认执行execute()方法 */
public String execute(){

System.out.println("当调用其它方法就不会走这个方法!");
return "";
}

/** 执行处理加法 */
public String add(){
this.fruit=num1+num2;//加
return "fruitPage";
}

/** 执行处理减法 */
public String subtract(){
this.fruit=num1-num2;//减
return "fruitPage";
}

/** JavaBean */
public int getNum1() {
return num1;
}

public void setNum1(int num1) {
this.num1 = num1;
}

public int getNum2() {
return num2;
}

public void setNum2(int num2) {
this.num2 = num2;
}

public int getFruit() {
return fruit;
}

public void setFruit(int fruit) {
this.fruit = fruit;
}

}


 

4. fruit.jsp响应结果的页面

<body>
<!-- 结果页面 -->
计算结果:<s:property value="fruit" />
</body>


 

1.2Action配置method属性(示列与以上代码大多一致,只修改有变更的):

  将Action类中的每一个处理方法都定义成一个逻辑Action方法。

1. index.jsp页面

<body>

<!-- Action配置method属性    使用:Struts2标签也可以使用HTML标签 -->
<s:form name="form" theme="simple" >

num1:<s:textfield name="num1" />
num2:<s:textfield name="num2" />
<s:submit type="button" value="加" onclick="computeMethod('addAction')" />
<s:submit type="button" value="减" onclick="computeMethod('subtractAction')" />
</s:form>

<!-- js -->
<script type="text/javascript">
function computeMethod(op){
document.form.action=op;//动态选择处理请求的方法
document.form.submit();//提交
}

</script>

</body>


 

2. struts.xml配置信息

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE struts PUBLIC "-//Apache Software Foundation//DTD Struts Configuration 2.1//EN" "struts-2.1.dtd" >
<struts>
<package name="struts2" extends="struts-default">
<action name="addAction" class="com.struts.ComputeAction" method="add" >
<result name="fruitPage" >/fruit.jsp</result>
</action>
<action name="subtractAction" class="com.struts.ComputeAction" method="subtract" >
<result name="fruitPage" >/fruit.jsp</result>
</action>
</package>
</struts>


  通过action元素的method属性来指定Action执行时调用的方法。

    优点:使得以更加安全的方式来实现动态方法的调用,不让别人看到你的实现方法。

    缺点:繁琐,一个处理请求的方法要跟一个action。

  Struts2根据method属性查找方法有两种途径:

        1.查找与method属性值完全一致的方法
        2.查找doMethod形式的方法
   使用动态方法调用和method属性的区别:

    1.通过以上三个struts.xml中的配置信息例子来说,他们的共同点是都在操作同一个Action。

    2.<form action="">中请求地址不同。

    3.动态方法的返回值相同,则会通过result进入一个页面。而method属性就算两个方法的返回值相同但进去不同的result,可能会进入两个不同的页面。

    由上可以分析出:

      (1)如果使用同一个Action,不同的处理请求的方法,响应使用相同的配置(result等)则使用动态方法调用。

      (2)如果使用同一个Action,不同的处理请求的方法,响应分别使用不同的配置,则使用action元素的method属性,为同一个Action配置多个名称。

      

 1.3使用通配符映射(wildcard mappings)方式([b]示列与以上代码大多一致,只修改有变更的):[/b]

1. index.jsp页面只改动了js部分。

<body>

<!-- 使用通配符映射(wildcard mappings)方式    使用:Struts2标签也可以使用HTML标签 -->
<s:form name="form" theme="simple" >

num1:<s:textfield name="num1" />
num2:<s:textfield name="num2" />
<s:submit type="button" value="加" onclick="computeMethod('addAction')" />
<s:submit type="button" value="减" onclick="computeMethod('subtractAction')" />
</s:form>

<!-- js -->
<script type="text/javascript">
function computeMethod(op){
document.form.action=op;//相比mothod属性改动只有这里
document.form.submit();//提交
}

</script>

</body>


 

 2.struts.xml的配置信息

  在使用method属性来实现同一个Action的不同方法处理不同的请求时,会发现,随着方法的增多,从而导致大量的Action配置,这时我们就需要通过使用通配符来解决Action配置过多的方法。

  在配置<action.../>元素时,需要指定name、class、method属性。其中name属性可支持通配符,然后可以在class、method属性中使用表达式。通配符用星号 * 表示。

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE struts PUBLIC "-//Apache Software Foundation//DTD Struts Configuration 2.1//EN" "struts-2.1.dtd" >
<struts>
<package name="struts2" extends="struts-default">

<action name="*Action" class="com.struts.ComputeAction" method="{1}" >
<result name="fruitPage" >/fruit.jsp</result>
<!-- <result name="fruitPage" >/{1}.jsp</result>表达式也可以写在这里 -->
</action>
</package>
</struts>


 

2.默认Action:
  在浏览器输入一个不存在的Action,页面将呈现404错误,为了网站更友好,我们可以设置一个默认的Action。
  
  注意:有一部份的朋友在某个自定义的action中定义default-action-ref这个配置的时候,认为在地址栏中输入地址如http://localhost:8080/project的时候(project为项目名),如果该项目后面不输入任何名字或者输错地址,则会自动进入default-action-ref定义的action并进入对应的类方法中进行操作并根据result返回页面,但是很多人发现结果并不是这样,而不管怎样都返回进入到index.jsp页面。
     实际上这一点从原理上来讲可以理解,default-action-ref这个配置的意思是当用户在点击了没有定义的action时,如果struts没有找到用户定义的action名称,则会自动跳转到该默认定义的action中。

    个人觉得地址栏中项目后不写名称和名称不存在是两个概念。

示列:

1. struts.xml 就在通配符例子中配置上默认Action

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE struts PUBLIC "-//Apache Software Foundation//DTD Struts Configuration 2.1//EN" "struts-2.1.dtd" >
<struts>
<package name="struts2" extends="struts-default">
<!-- 配置默认Action -->
<default-action-ref name="defaultAction"></default-action-ref>
<action name="defaultAction">
<result>/error.jsp</result>
</action>

<!-- 通配符映射(wildcard mappings) -->
<action name="*Action" class="com.struts.ComputeAction" method="{1}" >
<result name="fruitPage" >/fruit.jsp</result>
<!-- <result name="fruitPage" >/{1}.jsp</result>表达式也可以写在这里 -->
</action>
</package>
</struts>


2. index.jsp页面 这里我们把提交的url :Action地址链接,写错打断,当提交时找不到对应的Action,则会进入默认Action,进入error.jsp页面

<body>

<!-- 使用通配符映射(wildcard mappings)方式    使用:Struts2标签也可以使用HTML标签 -->
<s:form name="form" theme="simple" >

num1:<s:textfield name="num1" />
num2:<s:textfield name="num2" />    <!-- 测试默认Action,当提交的Action地址错误。则会走默认Action -->
<s:submit type="button" value="加" onclick="computeMethod('ssss')" /><!-- 把动态url地址乱写 -->
<s:submit type="button" value="减" onclick="computeMethod('subtractActios')" /><!-- 或在url地址中多加字符 -->
</s:form>

<!-- js -->
<script type="text/javascript">
function computeMethod(op){
document.form.action=op;//改动只有这里
document.form.submit();//提交
}

</script>

</body>


3. error.jsp 创建此页面查看效果

<body>
错误页面。!
未找到,Action实例时会默认走此页面!
</body>


 

3.[b]处理结果[/b]

  Struts2的Action处理完用户请求后,将返回一个普通字符串,整个普通字符串就是一个逻辑视图名。Struts2通过配置逻辑视图名和物理视图资源之间的映射关系,一旦系统收到Action返回的某个逻辑视图名,系统就会把对应的物理视图资源呈现给浏览者。

 
3.1    配置处理结果:
  Struts2的Action处理用户请求结束后,返回一个普通字符串-逻辑视图名,必须在struts.xml文件中完成逻辑视图和物理视图资源的映射,才可让系统转到实际的视图资源。
 
  Struts2通过在struts.xml文件中使用<result …/>元素来配置结果。Struts2提供了两种结果。
    局部结果:将<result …/>作为<action …>元素的子元素配置。
    全局结果:将<result …/>作为<global-results …>元素的子元素配置。
在package元素中配置<global-results>子元素:
    全局结果可满足一个包中多个Action共享一个结果:
    

<!-- 全局结果可满足一个包中多个Action共享一个结果,也就是说,当多个Action中都有一个重复的result时就可以使用全局结果,也就是说公共的result -->
<global-results>
<result name="fruitPage" type="dispatcher" >/fruit.jsp</result>
</global-results>


 
3.2.    处理结果类型:
  Struts2提供了对不同种类返回结果的支持,常见的有JSP,FreeMarker,Velocity等。
  Struts2支持的不同类型的返回结果为:[b](加粗为常用)[/b]
 
名字说明
chain用来处理Action链
dispatcher用来转向页面,通常处理JSP,这是默认的结果类型
freeMarker处理FreeMarker模板
httpHeader用来控制特殊的Http行为
redirect重定向到一个URL
redirect-action重定向到一个Action
stream向浏览器发送InputSream对象,通常用来处理文件下载
velocity处理Velocity模板
xslt处理XML/XLST模板
plaintext显示原始文件内容,例如文件源代码
tiles结合Tile使用
  另外第三方的Result类型还包括JasperReports Plugin,专门用来处理JasperReport类型的报表输出;Jfreechart Plugin;JSF Plugin。

  常用示列:

1.struts.xml

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE struts PUBLIC "-//Apache Software Foundation//DTD Struts Configuration 2.1//EN" "struts-2.1.dtd" >
<struts>
<package name="struts2" extends="struts-default">

<!-- 默认Action -->
<default-action-ref name="defaultAction"></default-action-ref>

<!-- 全局结果可满足一个包中多个Action共享一个结果,也就是说,当多个Action中都有一个重复的result时就可以使用全局结果,也就是说公共的result
<global-results>
<result name="fruitPage" type="dispatcher" >/fruit.jsp</result>
</global-results> -->

<action name="defaultAction">
<result>/error.jsp</result>
</action>
<!-- 通配符映射(wildcard mappings) -->
<action name="*Action" class="com.struts.ComputeAction" method="{1}" >
<!--1
表达式{1}也可以写在url连接中,class,name中都可以写,也可以写多少,索引从1开始
<result name="fruitPage" >/{1}.jsp</result> -->

<!--2
默认dispatcher转发跳转
<result name="fruitPage" type="dispatcher" >/fruit.jsp</result> -->

<!--3
重定向跳转
<result name="fruitPage" type="redirect" >/fruit.jsp</result> -->

<!--4
redirectAction:    Action实例 与另一个Action实例互相跳转
<result name="fruitPage" type="redirectAction" >skipAction</result> -->

<!--4.1
使用感叹号指定跳转方法,xml会显示报错,但可以用。    使用?&不能在Action实例中带参数
<result name="fruitPage" type="redirectAction" >skipAction!add</result> -->

<!--4.2
跳转Action带参数的方式:
actionName:跳转Action的名称
method:跳转Action实例中的哪个方法
num:带的参数,可写固定,可使用${属性名}取上一个Action实例中的属性,实现动态传值。
-->
<result name="fruitPage" type="redirectAction" >
<param name="actionName">skipAction</param>
<param name="method">add</param>
<param name="num1">${num1}</param>
<param name="num2">${num2}</param>
</result>
</action>

<action name="skipAction" class="com.struts.SkipAction" >
<result name="success" type="dispatcher" >/fruit.jsp</result>
</action>
</package>
</struts>


 

 2.再创建一个SkipAction 控制器的类

package com.struts;
/**
* 测试与dispatcherAction之间的传值
* @author asus
*
*/
public class SkipAction {

/** 接收computeAction实例传过来属性 */
private int num1;
private int num2;
private int result;//结果返回给页面

public String execute(){
System.out.println("当指定要走的方法时不会走此方法");
return "success";
}

/**添加的方法 */
public String add(){
result=num1+num2;
System.out.println("走add方法!");
return "success";
}

/** Get,Set方法 */
public int getNum1() {
return num1;
}

public void setNum1(int num1) {
this.num1 = num1;
}

public int getNum2() {
return num2;
}

public void setNum2(int num2) {
this.num2 = num2;
}

public int getResult() {
return result;
}

public void setResult(int result) {
this.result = result;
}

}
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: