宝宝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}&amp;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。

原创粉丝点击