struts2

作者: 罗志贇 | 来源:发表于2017-03-28 13:59 被阅读0次

    package元素

    属性:

    name:必须的,配置文件中要唯一.

    extends:指定父包.会把父包中的配置内容继承下来.一般需要直接或间接的继承一个叫做"struts-default"的包(在struts-default.xml配置文件中),如果不继承,那么struts2的核心功能将无法使用

    abstract:是否是抽象包.没有任何action子元素的package可以生命为抽象包

    namespace:指定名称空间.一般以"/"开头,该包中的动作访问路径:namespace+动作名称.如果namespace="",这是默认名称空间,和不写该属性是一样的

    屏幕快照 2017-01-17 12.51.17.png

    action元素

    作用:定义一个动作

    属性:

    name:必须的,动作名称,用户用户发起请求,在包中要唯一

    class:指定动作类的全名,框架会通过反射机制实例化,默认是:com.opensymphony.xwork2.ActionSupport.

    method:指定动作类中的动作方法.框架会执行该方法,默认是execute();

    关于默认类:是在struts-default.xml中进行了声明

    动作类

    1、编写动作类的三种方式:

    • POJO(Plain Old Java Object)普通的JavaBean
    public Hello{
      public String sayHello(){
        return "success";
      }
    }
    
    • 实现com.opensymphony.xwork2.Action接口
    public Hello implements Action{
      public String sayHello(){
        return SUCCESS;
      }
    }
    

    Action接口中的常量:

    String SUCCESS:success。一切正常。

    String NONE:none。动作方法执行后,不转向任何的结果视图。或者在动作方法中返回null。

    String ERROR:error。动作方法执行时遇到异常,转向错误提示页面。

    String INPUT:input。验证、转换失败,转向输入页面。

    String LOGIN:login。检测用户是否登录,没有登录转向此视图。

    • 继承com.opensymphony.xwork2.ActionSupport(推荐)

    意义:提供了一些基本的功能。比如验证和国际化消息提示等。

    public Hello extends ActionSupport{
    }
    

    Struts2的配置

    配置文件和加载顺序

    default.properties struts-core-**.jar org.apache.struts包中

    struts-default.xml struts-core-**.jar 包中

    struts-plugin.xml 插件中

    struts.xml 构建路径的顶端

    struts.properties 构建路径的顶端

    web.xml;

    在动作中访问ServeletAPI

    通过ServletActionContext静态方法

    ServletContext servletContext = ServletActionContext.getServletContext();
            HttpServletRequest request = ServletActionContext.getRequest();
            HttpServletResponse response = ServletActionContext.getResponse();
    

    Struts2提供的结果类型

    • dispatcher:转发到一个jsp
    • redirect:用于重定向到另外一个jsp页面
    • redirectAction:用于重定向到另外一个动作
    • chain:用户转发到另外一个动作
    • stream:用于文件下载

    自定义结果类型

    • 定义一个类继承StrutsResultSupport
    public class CaptachaResult extends StrutsResultSupport{
        
        //输出你的结果即可
        protected void doExecute(String finalLocation, ActionInvocation invocation)
                throws Exception {
            
        }
    
    }
    
    • 在package中定义结果类型
      • result-type 中 name:指定结果类型的名字,class:指定的结果类型包名
    <package name="p3" extends="struts-default">
            <result-types>
                <!-- 结果类型定义 -->
                <result-type name="captcha" class="com.itheima.results.CaptachaResult"></result-type>
            </result-types>
            <action name="captcha">
                <result name="success" type="captcha">
                    <param name="width">200</param>
                </result>
            </action>
        </package>
    

    封装参数

    方式一:动作类当做模型对象

    login.jsp

    <body>
        <form action="${pageContext.request.contextPath}/regist.action" method="post">
            用户名:<input type="text" name="username"/><br/>
            密码:<input type="password" name="password"/><br/>
            <input type="submit" value="登录"/>
        </form>
      </body>
    

    struts.xml

    <package name="p1" extends="struts-default">
            <action name="regist" class="com.itheima.action.UserAction" method="regist">
                <result>/success.jsp</result>
                <result name="login">/login.jsp</result>
            </action>
        </package>
    

    动作类UserAction

    //封装参数:方式一:动作类当做模型对象
    public class UserAction extends ActionSupport {
        private String username;
        private String password;
        
        public String getUsername() {
            return username;
        }
        //框架会按照表单的name值,调用对应的setter属性
        public void setUsername(String username) {
            this.username = username;
        }
    
        public String getPassword() {
            return password;
        }
    
        public void setPassword(String password) {
            this.password = password;
        }
    
        public String regist(){
            System.out.println(username+":"+password);
            if("wzhting".equals(username)&&"sorry".equals(password))
                return SUCCESS;
            else
                return LOGIN;
        }
    }
    
    

    方式二:动作和模型分开

    动作:

    public class PersonAction extends ActionSupport {
        private Person person = new Person();
    
        public Person getPerson() {
            System.out.println("getter");
            return person;
        }
    
        public void setPerson(Person person) {
            System.out.println("setter");
            this.person = person;
        }
        public String regist(){
            System.out.println(person);
            if("wzhting".equals(person.getUsername())&&"sorry".equals(person.getPassword()))
                return SUCCESS;
            else
                return LOGIN;
        }
    }
    

    模型:

    //属性和form中的字段名一致
    public class Person implements Serializable {
        private String username;
        private String password;
        public String getUsername() {
            return username;
        }
        public void setUsername(String username) {
            this.username = username;
        }
        public String getPassword() {
            return password;
        }
        public void setPassword(String password) {
            this.password = password;
        }
        @Override
        public String toString() {
            return "Person [username=" + username + ", password=" + password + "]";
        }
        
    }
    

    配置文件:

    <package name="p2" extends="struts-default" namespace="/person">
            <action name="regist" class="com.itheima.action.PersonAction" method="regist">
                <result>/success.jsp</result>
                <result name="login">/login1.jsp</result>
            </action>
        </package>
    

    jsp

    <body>
        <form action="${pageContext.request.contextPath}/person/regist.action" method="post">
            用户名:<input type="text" name="person.username"/><br/>
            密码:<input type="password" name="person.password"/><br/>
            <input type="submit" value="登录"/>
        </form>
      </body>
    

    方式三:实现ModelDriver

    public class CustomerAction extends ActionSupport implements ModelDriven<Customer>{
        //必须实例化
        private Customer customer = new Customer();
    
        public Customer getCustomer() {
            return customer;
        }
    
        public void setCustomer(Customer customer) {
            this.customer = customer;
        }
        public String regist(){
            System.out.println(customer);
            if("wzhting".equals(customer.getUsername())&&"sorry".equals(customer.getPassword()))
                return SUCCESS;
            else
                return LOGIN;
        }
        //此方法会在动作方法执行前先执行;得到当前对应的模型对象并把模型对象压入值栈的栈顶。setUsername先调用栈顶对象的该方法。
        //后面会有值栈(存放数据的大仓库)的详细介绍
        public Customer getModel() {
            return customer;
        }
    }
    

    Customer类略

    配置文件:

    <package name="p3" extends="struts-default" namespace="/customer">
            <action name="regist" class="com.itheima.action.CustomerAction" method="regist">
                <result>/success.jsp</result>
                <result name="login">/login2.jsp</result>
            </action>
        </package>
    

    jsp视图:

    <body>
        <form action="${pageContext.request.contextPath}/customer/regist.action" method="post">
            用户名:<input type="text" name="username"/><br/>
            密码:<input type="password" name="password"/><br/>
            <input type="submit" value="登录"/>
        </form>
      </body>
    

    OGNL表达式

    注意事项:必须要写在struts2的标签中

    • 支持对象方法调用:

      <!--s:property相当于c:out-->
      <s:property value="'abcdefg'.length()">
      
    • 支持静态的方法调用和值访问

      <!--访问静态变量-->
      <s:property value="@java.lang.Integer@MAX_VALUE" />
      <!--访问静态方法 配置参数 <constant name="struts.ognl.allowStaticMethodAccess" value="true"></constant>-->
      <s:property value="@java.lang.Math@abs(-100)" />
      
    • 访问OGNL上下文(OGNL context)和ActionContext;
    • 操作集合对象
    context上下文

    application:ServletContext中的所有属性attributes

    • 相当于EL的内置对象applicationScope

    session:HttpSession中的所有属性attributes

    • 相当于EL内置对象sessionScope

    value statck(根):是一个List

    action:动作类

    request:ServletRequest中的所有属性

    • 相当于EL的内置对象requestScope

    parameters:Map map = request.getParameterMap()

    • 相当于EL内置对象param Values

    attr :从四大域范围中搜索

    • ${}

    动作类的生命周期:每次访问都会重新创建心的动作类的实例.还会创建ActionContext和ValueStack实例.ActionContext和ValueStack一直保持在你的线程中(ThreadLocal)

    获取ActionContext的实例:

    ActionContext ac = ActionContext.getContext();

    往ContextMap存数据
    //向contextMap中存入数据
    context.put("contextMap","hello contextMap");
    
    //向HttpSession域范围内存入数据
    //第一种方式:使用contextMap的session
    Map<String,Object> sessionAttributes = context.getSession();
    sessionAttributes.put("sessionMap","hello sessionMap");
    //方式二:使用原始方式
    HttpSession session = ServletActionContext.getRequest().getSession();
    session.setAttribute("sessionMap1","hello sessionMap1");
    
    //向ServletContext域中存放数据
    //第一种方式,使用contextMap中的application(它也是一个Map)
    Map<String,Object> applicationAttributes = context.getApplication();
    applicationAttributes.put("applicationMap","hello applicationMap");
    //第二种方式,使用原始方式
    ServletContext application = ServletActionContext.getServletContext();
    application.setAttribute("applicationMap1","hello applicationMap1");
    
    
    往ContextMap存数据取数据
    <s:property value="#contextMap" />
    
    <s:property value="#session.sessionMap" />
    <s:property value="#session.sessionMap1" />
    
    <s:property value="#application.applicationMap" />
    <s:property value="#application.applicationMap1" />
    

    获取ValueStack的实例:

    方式一:

    ActionContext ac = ActionContext.getContext();
    ValueStack vs = ac.getValueStack();
    

    方式二:

    MaP<String,Object> requestAttributes = (MaP<String,Object>)ac,get("request");
    ValueStack vs = requestAttributes.get("struts.valueStack");
    

    方式三:

    ActionContext ac = ActionContext.getContext();
    ac.get(ValueStack.VALUE_STACK);
    

    ValueStack

    ValueStack实际上就是一个容器,每次前段页面发送一个请求时,就会将请求的数据进行封装,并入ValueStack的栈顶,发送一次请求开始,struts会为每一个action创建一个ValueStack

    ValueStack是一个接口,我们使用OgnlValueStack,贯穿action的整个生命周期

    ValueStack组成部分

    root区域:本质上就是一个ArrayList

    context区域:本质上就是一个map

    往root中存放数据
    valueStacke.set("","");此方法是操作根的

    检测栈顶是不是一个Map,如果不是,创建一个Map,Map中的数据就是p1=pp1,再把这个Map压入栈顶

    ValueStack vs = ActionContext.getContext().getValueStack();
    vs.set("p1","pp1");
    

    vs.put(Object obj);//放入一个对象

    ValueStack vs = ActionContext.getContext().getValueStack();
    vs.put(new User("name","age"));
    

    第一个参数是ONGL表达式,从栈顶开始搜索,调用setName("luoluo");

    ValueStack vs = ActionContext.getContext().getValueStack();
    vs.setValue("name","luoluo");
    

    向contextMap中存放数据,相当于ActionContext.getContext().put("abc","shit1");

    ValueStack vs = ActionContext.getContext().getValueStack();
    vs.setValue("#abc","shit1");
    

    参数是OGNL表达式,取数据

    ValueStack vs = ActionContext.getContext().getValueStack();
    Object obj1 = vs.findValue("name");
    
    ValueStack的其他方法
    setValue方法
    vs.setValue("name","张三");//就是把根中第一个name属性,设置为张三;
    
    vs.setValue("#name","李四");//就是向contextMap中放数据
    

    操作集合对象

    List:

    <s:radio name="gender" list="{'男','女'}" />
    

    Map:

    <s:radio name="gender1" list="#{'1':'男':'0':'女'}">
    

    iterator:

    <s:iterator value="students" var="s" status="vs">
        <s:property value="#s.name" />${s.name}---${s.age}
        <s:property value="#vs.index" />
    </s:iterator>
    

    OGNL中的特殊符号使用

    "#"作用:

    • 获取contextMap中的数据
    • 构建map集合

    "%"作用:

    • 如果想把一个普通的字符串强制看成时OGNL

      //因为'#'显示OGNL表达式的内容只能通过<s:property >来显示,其他的struts标签不管用,所以需要'%'
      
      <s:textfield value="%{name}" />
      

    "$":

    作用:在配置文件中使用,表示一个占位符

    struts2常用标签

    set标签:

    value存入map中属性的值,是一个OGNL表达式

    var存入map中属性的key

    scope:存入的范围有application,session,request,page和action

    如果不写,默认是action,它是在contextMap中和request范围内各存一份

    <!--在contextMap中不会有个名称叫str,值是abc的,因为abc是一个ognl表达式,在栈顶元素查找没有abc对应的值,再去contextMap中查找也米有,所以就不会存入任何内容-->
    <s:set value="abc" var="str" scope="session" />
    <!--把abc看成方式一个字符串,这个时候在session的Map中可以找到-->
    <s:set value="'abc'" var="str" scope="session" />
    <s:set value="'def'" var="str2" />
    
    <s:property value="#session.str" />
    <s:property value="#str2" />
    

    拦截器

    拦截器和filter的区别:

    拦截器只拦截action(方法),filter过滤任何请求

    struts执行流程

    请求—>StrutsPrepareAndExecuteFilter—>执行doFilter方法(判断请求的是action还是其他资源,是action继续往内部核心走,不是的话就直接放行)

    请求到达内部核心—>生成action的代理对象—>action执行类(执行一组拦截器)—>执行action中的指定的方法(查看是否有返回值,若有返回值到某个指定资源上,若没有方法执行完成)—>逆序执行一组拦截器—>StrutsPrepareAndExecuteFilter(放行之后的代码)—>生成响应信息,返回用户页面上

    相关文章

      网友评论

        本文标题:struts2

        本文链接:https://www.haomeiwen.com/subject/bpyjottx.html