三、Struts2的配置文件
1、加载时机:
当应用被tomcat加载的时候,struts2的配置文件就已经被加载过了。
2、加载顺序
顺序 |
配置文件名 |
所在位置 |
说明 |
1 |
default.properties |
struts2-core-2.3.15.3.jar\org\apache\struts2 |
不能修改 |
2 |
struts-default.xml |
struts2-core-2.3.15.3.jar |
不能修改 |
3 |
strtuts-plugin.xml |
在struts2提供的插件jar包中 |
不能修改 |
4 |
struts.xml |
我们的应用中 |
我们修改的:推荐 |
5 |
struts.properties |
我们的应用中 |
我们修改的 |
6 |
web.xml |
我们的应用中 |
我们修改的,可以给过滤器配置参数 |
注意:
1、Struts2提供了两种配置的方式。一种是key=value的方式,即使用.properties文件。另一种是xml文件配置。我们推荐使用xml文件(它能描述层级关系)。
2、当多个配置文件中,有相同的参数,后面的会把前面的值给覆盖了。
四、Struts2框架提供的常量
1、常用的常量
常量定义在了default.properties配置文件中,体现形式都是key=value。所有的struts2应用都会用到这些常量。
常用的:
常量名 |
常量值 |
说明 |
struts.i18n.encoding |
UTF-8 |
应用中使用的编码 |
struts.objectFactory.spring.autoWire |
name |
和spring框架整合有关 |
struts.multipart.parser |
jakarta |
指定文件上传用的组件 |
struts.multipart.maxSize |
2097152 |
文件上传总文件大小限制:2M |
struts.action.extension |
action,, |
能进入Struts2框架内部的url地址后缀名。多个值用逗号分隔 |
struts.devMode |
false |
是否是开发模式。开发模式:改了配置文件,不需要重启。输出更多的错误信息。开发阶段建议为true。 |
struts.ui.theme |
xhtml |
页面展示用的主题 |
2、在struts.xml中覆盖常量
使用<constant name="" value=""></constant>元素进行覆盖
例如:
<!-- 修改struts2常量使用的元素是constant
属性name:对应的key
value:对应的值
-->
<!-- 开启开发者模式 -->
<constant name="struts.devMode" value="true"></constant>
<!-- 修改struts2默认拦截的url后缀 -->
<constant name="struts.action.extension" value="do"></constant>
3、创建struts.properties文件覆盖
在应用的构建路径中创建一个struts.properties的文件。
struts.action.extension=abc
可以覆盖默认访问动作方法后缀
4、在web.xml中配置过滤器参数
<filter>
<filter-name>struts2</filter-name>
<filter-class>
org.apache.struts2.dispatcher.ng.filter.StrutsPrepareAndExecuteFilter
</filter-class>
<!-- 配置过滤器的初始化参数 -->
<init-param>
<param-name>struts.action.extension</param-name>
<param-value>html</param-value>
</init-param>
</filter>
<filter-mapping>
<filter-name>struts2</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>
五、xml配置文件的主要元素
1、package元素
1.1、作用:
在struts2的配置文件中引入了面向对象思想,使用了分包管理。易于管理动作类。便于模块化开发动作类。
1.2、属性:
name:包的名称。必须写。且必须唯一。
extends:一般情况下需要继承struts-default包,但不是必须的。不过如果不继承的话,将无法使用struts2提供的核心功能。
struts-default.xml中定义着struts-default这个包。而struts-default.xml是在我们的struts.xml加载之前加载。
abstract:把包声明为抽象包,抽象包就是用来被继承的。只要是没有<action>元素的包,就可以声明为抽象包。
namespace:名称空间。
名称空间 + 动作名称 = 访问路径
<package name="p1" extends="struts-default">
<!-- action元素的name属性只需要写页面上提供hello.action前面的部分。后缀不需要写,也不能写。-->
<action name="hello" class="cn.itcast.web.action.HelloAction" method="sayHello">
<!-- 执行完成之后,还要响应浏览器
name属性的取值是和动作方法的返回值一致,且严格区分大小写。
-->
<result name="success" type="dispatcher">/success.jsp</result>
</action>
</package>
1.3、packege中的namespace
namespace的默认值:
a.不写该属性
b.写了该属性,取值是一个"".
注意:默认值不是/
动作类的搜索顺序:
先找名称空间,再找动作名称。
<!--
package元素:
作用:把动作类都分包来管理。目的是便于我们更好的模块化开发。配置文件中引入了面向对象的思想。
属性:
name:指定包的名称。该属性必须写。名称取值要唯一。
extends:指定当前包的父包。子包自动具备父包定义的内容。一般我们都继承struts-default的包,该包定义着struts2的核心功能。
struts-default包在struts-default.xml配置文件中定义着。如果不继承struts-default包,struts2核心功能都用不了。
abstract:把当前包声明为抽象包。抽象包一般都定义公共的配置,而且就是用来被继承的。
注意:只有没有action元素的包,才能被声明为抽象包。
namespace:指定当前包的名称空间。模块化管理访问URL。
当我们配置了名称空间之后:
访问URL就变成了 名称空间+hello
struts2框架提供的名称空间,在不写的时候是有默认值得。
默认值是:""
"/"不是默认的名称空间,而是根名称空间。
-->
<package name="p1" extends="struts-default" namespace="/n1">
<action name="hello" class="cn.itcast.web.action.HelloAction" method="sayHello">
<result name="success" type="dispatcher">/success.jsp</result>
</action>
</package>
2、action元素
2.1、作用:
配置动作用的。
2.2、属性:
name:动作名称
class:动作类全名。默认的动作类是:com.opensymphony.xwork2.ActionSupport
是在struts-default.xml中定义的
method:动作类中的方法名称。默认是public String execute(){}
要求:
1.public的
2.返回值必须是String
3.没有参数
2.3、动作类
a.方式一:动作类就是一个POJO,非常简单的javabean。
b.方式二:动作类实现com.opensymphony.xwork2.Action接口。
常量:给动作方法返回值用的。用常量可以使你的应用规范和统一。
常量变量名 |
对应的值 |
说明 |
SUCCESS |
success |
动作方法执行一切OK |
ERROR |
error |
动作方法执行时遇到了异常 |
INPUT |
input |
回显 |
LOGIN |
login |
一般转向登陆页面 |
NONE |
none |
不转向任何的视图 |
c.方式三:动作类继承com.opensymphony.xwork2.ActionSupport 推荐使用
2.4、动作的访问
a.使用通配符:
<package name="p1" extends="struts-default" namespace="/n1">
<!--
action元素:
作用:配置动作名称和动作类以及动作方法的对应关系。
name:指定的是动作名称。它是与页面上请求URL后面对应起来的。但是此处不要写后缀,也不能写后缀。
动作名称的要求:必须唯一
class:指定的是动作类的全类名。
method:指定的是动作方法名称。
动作类的三种定义方式:
第一种定义方式:无侵入式的
定义一个普通类,提供一个符合要求的动作方法。
动作方法的要求:
1、public
2、返回值是String
3、没有参数
-->
<action name="hello" class="cn.itcast.web.action.HelloAction" method="sayHello">
<result name="success" type="dispatcher">/success.jsp</result>
</action> <!-- 动作类的第二种定义方式:
实现Action接口。该接口中提供了一些常量
SUCCESS:一般多用于全部成功。
ERROR:当我们的动作方法运行出现执行异常时使用。
INPUT:当我们提交数据失败时,需要回显数据,一般使用INPUT。
LOGIN:一般返回登录界面。
NONE:不返回任何结果视图。一般多用于文件下载,输出json数据。
可以使用return null;来实现和return NONE一样的效果
-->
<action name="hello1" class="cn.itcast.web.action.Hello1Action" method="sayHello">
<result name="none" type="dispatcher">/success.jsp</result>
</action> <!-- 默认的动作方法:是Action接口中的execute方法。如果是执行该方法,可以不指定method属性。 -->
<action name="helloAction" class="cn.itcast.web.action.Hello1Action">
<result name="none" type="dispatcher">/success.jsp</result>
</action> <!-- 动作类的第三种定义方式:推荐的方式
继承ActionSupport的类,该类就是Action接口的实现类。
在我们的实际开发中:
一般动作类都要继承ActionSupport。
-->
<action name="hello2" class="cn.itcast.web.action.Hello2Action" method="sayHello">
<result name="success" type="dispatcher">/success.jsp</result>
</action> <!--
默认的动作类:ActionSupport
默认的动作方法:execute()
-->
<action name="defaultAction">
<result name="success" type="dispatcher">/success.jsp</result>
</action>
</package> <!--最基本版配置:
<package name="user" extends="struts-default" namespace="/user">
<action name="addUser" class="cn.itcast.web.action.UserAction" method="addUser">
<result name="success" type="dispatcher">/addUser.jsp</result>
</action>
<action name="updateUser" class="cn.itcast.web.action.UserAction" method="updateUser">
<result name="success" type="dispatcher">/updateUser.jsp</result>
</action>
<action name="deleteUser" class="cn.itcast.web.action.UserAction" method="deleteUser">
<result name="success" type="dispatcher">/deleteUser.jsp</result>
</action>
<action name="findUser" class="cn.itcast.web.action.UserAction" method="findUser">
<result name="success" type="dispatcher">/findUser.jsp</result>
</action>
</package> --> <!-- 使用*通配符来匹配一个或者多个字符
{n}表明的是通配符出现的位置。从1开始。 <package name="user" extends="struts-default" namespace="/user">
<action name="*" class="cn.itcast.web.action.UserAction" method="{1}">
<result name="success" type="dispatcher">/{1}.jsp</result>
</action>
</package>--> <!-- 使用通配符的升级版本
所有的action,都适用于此配置。它是通配符的形式。
我们还可以写成全字符匹配,或者多字符匹配。
他们之间有优先级问题:
全字符匹配最高>多字符匹配>全使用通配符
-->
<package name="user" extends="struts-default" namespace="/user">
<action name="*_*" class="cn.itcast.web.action.{2}Action" method="{1}{2}">
<result name="success" type="dispatcher">/{1}{2}.jsp</result>
</action> <action name="add_User" class="cn.itcast.web.action.UserAction" method="deleteUser">
<result name="success" type="dispatcher">/success.jsp</result>
</action>
</package>
六、结果类型视图(逻辑结果视图)
前奏:该部分内容指的就是struts配置文件中的result元素的使用
1、result元素
作用:
为动作指定结果视图
属性:
name:逻辑视图的名称,对应着动作方法的返回值。默认值是success。
type:结果类型,指的就是用什么方式转到定义的页面。默认是dispatcher。
2、result元素中type的取值
type属性的取值在struts-default.xml中定义着。
常用结果类型介绍:
dispatcher:(默认值)使用请求转发,转向一个页面。
redirect:使用重定向,转向一个页面。
redirectAction:注意:使用的是重定向。
a.重定向到另一个相同名称空间的动作。
b.重定向到不同名称空间的动作
3、result元素中param子元素
在转发或者重定向到不同包下的动作时,都用到了result元素的子元素param。
param元素的作用:依赖注入(Dependence Injection)思想
我们通过struts-default.xml中的resultTypes元素中配置可以看出,每个结果类型视图其实都是靠一个类来实现的。而param元素就是将配置的参数,注入到该类中。
调用的是对应类的setter方法进行注入的。
例如:redirectAction结果视图
该类中肯定会有对actionName和namespace属性的注入方法(setter方法)。
4、全局视图和局部视图
4.1、局部视图
只能当前动作使用
4.2、全局视图
所有动作都可以使用
注意:优先级问题,先局部后全局。
七、在动作类中访问Servlet的API
使用ServletActionContext类
public class HelloAction extends ActionSupport{ public String sayHello(){
HttpServletRequest request = ServletActionContext.getRequest();
HttpServletResponse response = ServletActionContext.getResponse();
HttpSession session = request.getSession();
ServletContext application = ServletActionContext.getServletContext();
System.out.println(request);
System.out.println(response);
System.out.println(session);
System.out.println(application);
return "success";
}
}
八、分文件编写框架配置文件
1、不分文件开发可能产生的问题
就类似于我们在写java类时,所有代码都写在一个类里,甚至写在一个方法里。
当3个人都checkout了struts.xml文件时,第一个人提交了,后面的人在没有更新就提交时,第一个人写的可能就白写了。
2、分文件编写Struts2的配置文件
<struts>
<!-- 开启开发者模式 -->
<constant name="struts.devMode" value="true"></constant> <package name="myDefault" extends="struts-default" abstract="true">
<global-results>
<result name="success">/success.jsp</result>
</global-results>
</package> <!-- 导入其他配置文件 -->
<include file="struts-book.xml"></include>
<include file="struts-user.xml"></include>
</struts>
<struts>
<!-- 与用户相关的动作配置 -->
<package name="user" extends="myDefault" namespace="/user">
<action name="addUser" class="cn.itcast.web.action.UserAction" method="addUser"></action>
</package>
</struts>
<struts>
<!-- 与图书相关的动作配置 -->
<package name="book" extends="myDefault" namespace="/book">
<action name="addBook" class="cn.itcast.web.action.BookAction" method="addBook"></action>
</package>
</struts>