宝宝Struts2学习总结
来源:互联网 发布:2017映客刷粉丝软件 编辑:程序博客网 时间:2024/05/01 11:14
这周花了一个星期的时间学习了Struts2,今天就总结一下,以便后面使用时可以参考,顺便理理思路,嘿嘿。
Struts和Struts2都是属于控制转发的框架,而在Struts2中功能比较强大,而且使用方便,不过我在好多测试中都很费神,因为不报错,也不出来我想要的结果,很无奈,嘿嘿!
首先说下它的应用如何搭建:在老版本中有5个必须包,但是现在我用的是2.1.8这个版本就不止5个jar包了,
包括:
struts2-core-2.1.8.jar(这是struts2的核心包),commons-loging-1.0.4.jar(通用日志记录包),
freemarker-2.3.15.jar(是一个模板引擎,一个基于模板生成文本输出的通用工具),
ognl-2.7.3.jar(这是struts2的表达式语言库即对象图导航语言),
xwork-core-2.1.6.jar(由于struts2就是从xwork衍生过来的,所以这包夜是必须的), commons-fileupload-1.2.1.jar(struts2的上传下载包),
common-io-1.3.2.jar(这个包在低版本中是不需要的),
struts-convention-plugin-2.1.8.jar(这是注解配置用的),我在这列出了这8个jar包基本是就够用了。
导入包后如果不用注解配置,那struts2也肯定有个配置文件,默认叫struts.xml,这个文件必须放在classes中,也就是放在src下。
这两件事做完后就要在web.xml中进行配置,因为struts2跟1不一样,它不用配置servlet,而是过滤器,配置如下(固定写法):
<filter>
<filter-name>struts2</filter-name>
<filter-class>org.apache.struts2.dispatcher.FilterDispatcher</filter-class>
<!-- 这个参数是注解用的 -->
<init-param>
<param-name>actionPackages</param-name>
<param-value>com.mwt.action.zhujie</param-value>
</init-param>
</filter>
<filter-mapping>
<filter-name>struts2</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>
以上步骤做完之后,struts2的架子也就搭起来了,好像不是很复杂哦!
接下来我主要说下struts2的运行流程和它的重点也就是常用的功能:
当我们在页面发出一个请求之后,由于服务器提前已经加载了web.xml,所以这个过滤器就会拦截到所有的请求,然后会默认去classes下的struts.xml中找对应的action(即action配置标签中的name属性),在struts2中访问后缀是.action,不写后缀也会被默认加上.action。action执行完后都会返回一个字符串,此时在去struts.xml中找result标签的name属性对应的资源,可以是jsp、html、.action等。
举例说明:
1、action
struts2中的action很简单,不用集成任何类,也不用实现任何接口,这和struts1是有区别的,他只要有一个返回字符串的execute方法就行了,但如果有多个方法时,名字也可以随便起,不一定是execute。
2、struts.xml
这个配置文件是struts2的核心配置文件,这个没什么可说的,就直接贴出代码再代码中详细的介绍吧:
<?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">
<!-- struts2的默认过滤编码是utf-8,所以我们也就不用去写过滤器来转码了,但是如果想改的话也可以 -->
<!-- struts2的配置文件要放在classes下,所以在项目中也就是放在src中了 -->
<struts>
<!-- 当一个项目中action非常多是我们就可以分开好多个xml配置文件然后用include去引入它们 -->
<include file="struts1.xml"></include>
<!-- 这个struts-default必须继承,它会默认加上好多拦截器,如果不继承它的话就连result等好多标签都无法使用 -->
<package name="bao" extends="struts-default">
<!-- 没有配置命名空间namespace,如果页面发出任何请求的话,那么struts会在这里的找所有的action中与name对应的action类,name是要访问的action起的名字,class对应的全类名 -->
<action name="login" class="com.mwt.action.LoginAction">
<result name="success">/success.jsp</result>
<!-- 在result中有个type属性就是专门配置是否为客户端跳转,默认是服务器端跳转,若加上redirect后浏览器地址栏会从客户端从新访问即就不会传递参数给页面了,就和Servlet中的sentRedirect一样 -->
<result name="error" type="redirect">/fail.jsp</result>
</action>
<action name="getRequest" class="com.mwt.action.GetRequestTest">
<!-- 跳到命名空间为global下的a2这个action -->
<result name="success" type="chain">/global/a2</result>
</action>
</package>
<!-- 在这说说package中的name属性,到现在为止我是没有感觉到它的意义, 但是我的一个朋友(秉承)说是当extends继承包时就是参考的这个name属性,看来这点是被我忽略了。-->
<package name="test2" extends="struts-default" namespace="/test2">
<action name="m" class="com.mwt.action.Method">
<!-- 这里调用不同的方法,在页面用感叹号动态调用,下面在用method做测试 -->
<result name="add">/methodTest/add.jsp</result>
<result name="update">/methodTest/update.jsp</result>
<result name="delete">/methodTest/delete.jsp</result>
</action>
</package>
<package name="test1" extends="struts-default" namespace="/test1">
<!-- 这里调用不同的方法用method做测试,缺点是同一个action中的不同方法调用要配置好多action,优点可以加不同的拦截器 -->
<action name="add" class="com.mwt.action.Method" method="add">
<result name="add">/methodTest/add.jsp</result>
</action>
<action name="update" class="com.mwt.action.Method" method="update">
<result name="update">/methodTest/update.jsp</result>
</action>
</package>
<!-- 通配符 (使用通配符时struts2会按照你在xml文件中配置的顺序去一一匹配),这样就可以只写一个action,只要方法和页面名字有规律就可以自动调用所有的方法和页面-->
<package name="User" extends="struts-default" namespace="/user">
<action name="*User" class="com.mwt.action.UserAction" method="{1}User">
<result name="success">{1}User.jsp</result>
</action>
</package>
<!-- action之间的跳转 -->
<package name="aciton1" extends="struts-default" namespace="/action1">
<action name="a1" class="com.mwt.action.A1">
<!-- 这个type="chai"指的是服务器端action的跳转,如果在不同的命名空间下就在a2前面加上另一个action的命名空间就行了,这里的param属性也可以不加 -->
<result name="success" type="chain">
<param name="actionName">a2</param>
</result>
<!-- <result name="success" type="chain">a2</result>也可以这么写 -->
</action>
<action name="a2" class="com.mwt.action.A2">
<!-- 在struts2中的配置文件中也可以用${}表达式,这个表达式很想EL表达式,但其实这是struts2自带的OGNL表达式(对象图导航语言),它会把前面action中的有get/set方法的属性值自动取出来 -->
<result name="success" type="redirect">/success.jsp?username=${username}</result>
</action>
</package>
<!-- 全局结果集,在下面的包中所有的action都会跳到同一个页面 -->
<package name="global" extends="struts-default" namespace="/global">
<global-results>
<result name="success">/success.jsp</result>
<result name="error">/fail.jsp</result>
</global-results>
<action name="a1" class="com.mwt.action.A1"/>
<action name="a2" class="com.mwt.action.A2"/>
</package>
<!-- action之间传递值(这个服务器端的传递会把第一个action中同名的属性值直接赋给第二个action中,他两都共用了同一个值栈,而且后放进去的(s2)先取出来。),如果是客户端跳转页可以用***.action?username=${name}&passward=${pwd}这中方式进行传值-->
<package name="student" extends="struts-default">
<action name="s1" class="com.mwt.action.Student1">
<result name="success" type="chain">s2</result>
</action>
<!--
自定义拦截器只要实现接口,或者继承一个类,这都是struts2自带的,如下配置。
<interceptors>
<interceptor name="a" class="b"></interceptor>
</interceptors>
-->
<action name="s2" class="com.mwt.action.Student2">
<result name="success">/success.jsp</result>
<!--在这里引用下上面配置的拦截器就行了(切记还要引用下默认的拦截器,否则struts2的好多功能就无效了,这就好比Java中的构造器一样。) <interceptor-ref name="a"></interceptor-ref> -->
</action>
</package>
</struts>
3、在看看action,这个action很简单就一个方法。
public class LoginAction extends ActionSupport{
User user;
public User getUser() {
return user;
}
public void setUser(User user) {
this.user = user;
}
public String execute(){
if(user.getUsername().trim().equalsIgnoreCase("baobao"))
return SUCCESS;
//因为继承了ActionSupport,所以可以直接这样返回,其实内部也是返回了一个小写的success字符串。
return ERROR;
}
}
4、在看看页面的访问方式:
<a href="<%=path%>/test2/m!add.action">add</a> 访问命名空间为test2下的name值为m的这个action种方法名叫add()的方法。
<a href="s1.action">s1</a><hr>
<a href="s1">s1</a><hr>这两中写法是一样的。
5、页面获取值方式:
<!--struts2的核心技术就是ValueStack值栈技术! 默认所有的action都会放在值栈中,一个请求就会产生一个值栈。栈是先进后出,如果是服务器之间action跳转(a1->a2的跳转再到页面),那么这两个action共用同一个值栈,而在前台取出的值是先从a2中找,然后从a1中找,栈是后来的在栈顶。值栈中的值不会在action中相互传递,同名属性值也不会被覆盖,都是遵循后进先出的形式去获取内容。 -->
<h1>欢迎您:<s:property value="user.username"/></h1>
<p>用户名:${param.username}</p><hr>
<p>Request作用域测试:username=${username}</p>
<!-- 在action中有get和set方法的属性在这里用值栈技术都能直接取出来 -->
<p>s=${s}</p><hr>
<!-- OGNL表达式测试(对象导航图语言) -->
ognl获得的用户名称:<s:property value="user.username"/><br>
用el也可以代替上面的struts2标签(说明struts2的值栈中的内容也放进了request作用域!):${user.username}<br>
<p><%=request.getAttribute("user.username")%></p><!-- 这里就证明了值栈中的值确实放到了request作用域中 -->
<!-- 在这里调用方法传递参数的时候,用单引号,像这样调用方法就可以替代自定义函数标签库了 -->
调用user对象中的公共方法(属于值栈中):<s:property value="user.f('ma','wentao')"/><br>
<!-- 如果调用action中或者JDK的静态方法就直接写全类名(试验未成功,很纳闷) -->
调用JDK中静态方法(返回个0到1的随机数):<s:property value="@java.lang.Math@random()"/><br>
如果是java.lang.Math下的类可以省略掉前面的类:<s:property value="@@random()"/><hr>
<!-- 很纳闷,测试失败,这个f()方法一直没有被执行。 -->
action中静态方法调用:<s:property value="@vs@f()"/><!-- 这里的vs就代表valuestack值栈,属于固定写法。 -->
6、在action中如何获得HttpServletRequest对象,有了它HttpSession也就很容易获得了。
/*
* 第一种方式:用ServletActionContext这个类的getRequest()静态方法就可以直接获取request,但必须在方法中 * 初始化,不能直接给类成员赋值。
* 第二种方式:用IOC控制反转直接注入request,这是struts2自带的注入,只要实现了ServletRequestAware这个借 * 口再实现了它的set方法就行了
* 今天就用第二种方式做个试验
* 注意:其实在用struts2时很少用到request这个对象,因为在action中所有的属性都会通过struts2的值栈技术直接 * 传到页面或者另一个action。
*
*/
private HttpServletRequest request;
public void setServletRequest(HttpServletRequest request) {
this.request = request;
}
//不用request,就用${}再页面直接取值,这种值栈技术很难让人琢磨
//s或者其他域只要有get和set方法就能在令一个action或者页面直接取值。
private String s;
private double d;
public String execute(){
request.setAttribute("username","感冒快点好吧!");
s="值栈技术存值";//在页面直接用${s}取值
d=1232354;//在另一个action中取值
return SUCCESS;
}
7、最后在说下struts2的注解,注解的好处显而易见,它可以让框架完全脱离xml配置文件,很容易学习,配置起来也相当方便,我还是挺喜欢这种配置的,其实好多框架都用到了注解,比如hibernate、spring都有注解,这些就跟JDK中的注解一样,比如@Override这个就是重写提示的注解。
我用struts2就做了一个小例子,可结果很令人失望,一直不出效果,难道与版本有关系?哎,还是先贴出代码吧!
/*
* struts2注解(先看action注解,这个struts2真的很奇怪,老版本跟新版本的注解变化很多)
* 老版本直接用,而且注解标示也不多,action也就4个,而这新版本我用的是struts2.1.8还要再导个包
* 叫struts2-convention-plugin-2.1.8.jar幸好这包在源码包中有。
*/
//首先标注必须在web.xml中配置注册下
//@ParentPackage(value="struts-default")默认就这包可以不写
//@Namespace(value="space")可以不要命名空间
//@Result(name="success",location="/success.jsp")这里只有一个跳转结果时用,在老版本这个location要换成value
@Namespace(value="space")
@Action(value="test",results={
@Result(name="success",location="/success.jsp"),
@Result(name="error",location="/fail.jsp",type="redirect")
})
public class ZhujieAction extends ActionSupport{
//这里的名字可必须以Action为后缀,访问时就以前面的单词首字母改成小写就行了。(不过有多个方法怎么分辨还不清楚,可能会在@Action()中配置多个value属性。)
public String test(){
if(1>0) return SUCCESS;
else return ERROR;
}
}//现在还是测试不成功,算了,先放弃吧!代码绝对没问题(多么自信的男人呀,嘿嘿!)。
好了,Struts2就先写到这,这些都是比较常用的知识点,其实struts2还有好多功能呢,比如自带了好多拦截器跟过滤器作用差不多,不过功能比较强大,还有它的好多标签在页面我也很少用,我从来不专门学习一个框架的标签的,感觉很浪费时间,而且意义不大,struts2还有自己的文件上传下载、异常处理、OGNL导航表达式等等,这些就等用到了再学吧。好了就先说到这吧,今天上班来的早,公司就我一个人,太安静了,所以突然说了这么多,嘿嘿!
补充:
★Struts中的OGNL(对象图导航语言)采用了struts2的标签获取值,因此我很少用,上面也说过不喜欢用某种框架自带的标签,但见了要能认识。比如:Struts2中的HttpSession已经被封装成了Map类型,只要类实现了SessionAware这接口,重写了setSession就可以自动注入进去,在页面用的话就直接 <s:property value="#session.userName"/>,这只是#的其中一种用法,其实它也就等价于session.getAttribute("userName");不过我还是习惯用HttpSession。
- 宝宝Struts2学习总结
- 宝宝 javaweb 学习总结
- 宝宝Hibernate学习总结
- 宝宝spring学习总结
- 宝宝Oracle学习总结
- 宝宝汇编学习总结
- 宝宝spring学习总结
- 宝宝进制转换学习总结
- [Struts2]Struts2学习总结
- Struts2之struts2学习总结
- Struts2 学习总结
- struts2学习总结
- Struts2学习知识点总结
- Struts2.0学习总结
- Struts2学习总结_Action
- Struts2开发学习总结
- Struts2 学习总结
- struts2框架学习总结
- 带滚动条的div
- GPRS A/Gb 模式下的移动管理状态介绍 (A/Gb mode Mobility Management States)
- 分页存储过程
- Posix线程编程指南(2)--线程私有数据
- 电路呼叫规程概念图 (CS Paging Procedure in A/Gb and Iu mode)
- 宝宝Struts2学习总结
- 移动台的GPRS 类型 ( MS's GPRS Class Type)
- 构建基本的Linux C 编程环境 (转的怎么收藏不了)
- cvs add bin
- CVCS & DVCS
- ti-idf算法,实现对英文文档的检索,把多篇文档中的词(英文单词),按照权值从小到大进行排列
- oracle数据库同步技术,高级复制
- 最小生成树的Kruskal算法(C/C++)--附有测试数据和结果
- ESock_Clien 14 原因分析