• 2.Struts2-Action


    struts.xml文件中 action 标签中几个属性的作用

      1.name:为action命名,输入url访问时,需要带上action的name,通过name知道访问哪个action(通过class属性)

      2.class:指定需要执行的action的类名,如果不指定,会使用struts2默认的一个action,同样会返回某个字符串结果

    自己的Action类的书写有三种方式:

    在不指定调用什么方法的情况下,默认会调用 访问的Action的 execute方法 ,

    也是通过这个 execute方法的返回值来确定 跳转到哪个页面

      1.自己编写 execute方法,返回某个字符结果集(如 "success")

    1 public class IndexAction1 {
    2     public String execute() {
    3         return "success";
    4     }
    5 }

       2.实现Action接口,因为是接口,里面没有方法的实现,所以都必须自己去实现这些方法

    1 public class IndexAction2 implements Action {
    2     @Override
    3     public String execute() {
    4         return "path";
    5     }
    6 }

      3.继承ActionSupport类,好处:ActionSupport中封装了很多方法可以直接使用  /*一般使用第三种*/

    1 public class IndexAction3 extends ActionSupport {
    2     @Override
    3     public String execute() {
    4         return "success";
    5     }
    6 }

    注:Action 中 execute 的方法的返回结果(String)和 struts.xml 中的 result 中的nae 属性相对应

    访问Action的过程:0.在前面struts2访问过程的基础上(之前没有写Action)

             1.请求访问某个action(根据该action 中的 name属性)

             2.根据action 的name ,得到action的class,通过该class /*new出一个IndexAction*/

             3.执行该action的execute方法,返回某个字符串结果与result中的name属性进行匹配,

              如果是匹配的话,就转发到result包含的页面进行显示

    注:struts2和 struts1 的一个重要的区别就是:struts2 每次访问这个Action都会创建一个Action对象 (虽然比较消耗内存,但是可以避免并发问题)

      而在struts1中,IndexAction 只会被创建一次(所以会有并发问题)

    动态方法调用

    Action 执行的时候不一定要执行execute方法,可以在配置文件中配置 action 的 method属性 来指定执行Action中的哪个方法

    但是这样一个方法就要配置一个Action,会配置太多的 Action,所以不好

    也可以在url 地址中动态指定(动态方法调用 DMI):

      如:<a href="<%=context %>/user/user!add">添加用户</a>  调用userAction的add方法

        <a href="<%=context %>/user/user!delete">删除用户</a>  调用userAction的delete方法 

    通配符

    使用通配符,可以将配置量降到最低(减少struts.xml中的配置)

    不过,一定要遵循 "约定优于配置"的原则(即 通配符使用的前提是项目人员对命名等有相同的约定)

    例: 

      <action name="*_*" class="com.bjsxt.struts2.wildcard.action.{1}Action" method="{2}">

        <result name="success">/{1}{2}Success.jsp</result>

      </action>

      <a href="<%=context %>/actions/Teacher_Delete">删除老师</a>,当点击这个超链接时 

      第一个* 代表了Teacher  第二个*代表了 Delete  

      (这样设置的话,struts.xml基本就不需要修改了,只需要修改Action 和 JSP显示页面就可以;)

      

    使用Action 接收参数的三种方式

      1.使用action 属性接收参数:1.在action中增加需要接收的属性(get set方法) 2.通过地址栏的输入为属性赋值

      2.使用Domain Model(或DTO)接收参数:

        1.创建一个User实体

        2.在action中把User当做属性(不需要new),为其设置set get方法

        3.通过地址栏的输入为属性赋值(需要指定model 如:user.name=xzk)

      过程应该是这样的:请求发送,找到对应的Action, 请求里面有user.name=xzk,就去找user,发现没有new, struts2就帮他new一个

      new 完之后,就往里面设值(set方法),注意地址栏中的对象的名称必须和Action中对象的名称保持一致,否则struts2都不知道去 new 谁

      3.使用ModelDriven接收参数

        1.创建一个User实体

        2.让Action 实现 ModelDriven 接口,重写getModel()方法

        3.在Action创建一个User实例对象(需要new出来),用于接收前台传来的参数

      Action实现ModelDriven 后的访问过程:

        1.拿到Action的class后,new一个action

        2.询问该Action是否实现了ModelDriven接口

        3.如果实现了,调用getModel方法,拿到Model(User)对象(action中创建的实例对象)

        4.根据这个user,完成 get set方法

      如果没有实现ModelDriven,则struts会自己new 一个,然后去get set

      这也是为什么使用Domain Model接收参数 只需要private User user; 而不需要new的原因(当然new 也可以)

    注:不一定要通过地址栏(get方法)来符属性赋值,可以使用Post方法,直接将表单提交到对应的action,只要action中有相应属性

    字符编码问题:想接收传来的中文数据,需要在struts.xml文件配置

      <constant name="struts.i18n.encoding" value="GBK" />

    (注:struts2.1.6 有Bug ,仍然无法解决编码问题)为了完成实验,需要 修改 web.xm文件 使用之前版本的过滤器

    简单数据校验

      1.发送数据给struts,通过struts.xml 找到 需要new 哪个action,将数据给这个action去处理

      2.如果传递过来的数据符合要求,则返回SUCCESS等,跳转相应页面

      3.如果数据不符合要求,则需要记录错误信息(this.addFieldError("error", "name is error");)

      

      4.在error 处理页面,取出错误信息并显示 , 需要导入 struts的标签库

        

     1 <?xml version="1.0" encoding="GB18030" ?>
     2 <%@ page language="java" contentType="text/html; charset=GB18030"
     3     pageEncoding="GB18030"%>
     4     <%@taglib uri="/struts-tags" prefix="s" %>
     5 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
     6 <html xmlns="http://www.w3.org/1999/xhtml">
     7 <head>
     8 <meta http-equiv="Content-Type" content="text/html; charset=GB18030" />
     9 <title>Insert title here</title>
    10 </head>
    11 <body>
    12     <s:fielderror name="error"></s:fielderror>
    13     <s:property value="errors.error[0]"></s:property>
    14     <s:debug></s:debug>
    15 </body>
    16 </html>

        <s:fielderror name="error"></s:fielderror>    (显示 name 为 error 的Map所有数据)

        <s:property value="errors.error[0]"></s:property>

        (这个value栈的名字叫 errors,存放所有错误信息的栈 显示 name 为 error 的Map的第一个数据

      注:在使用 this.addFieldError(",")后,数据是存放在 'value栈(name是String value是Map)' 中,

        可以通过 <s:debug></s:debug> 生成的链接查看

        s:fielderror 和 s:property的区别:1.一个默认取出全部数据,一个单独取出某个数据

        2.s:fielderror得到的数据会按照struts默认的格式显示(不常用) s:property直接得到的 就是一段字符串

    Action 中访问 web元素的方法(一)(如:取得Map类型request,session,application)

      1.在Action中,通过ActionContext.getContext()方法得到 ActionContext 对象,

      ActionContext对象 作用:得到Action执行的环境,上下文, ActionContext中有Web元素

      2.通过ActionContext对象,得到想要的web元素,

      

      3.为web元素添加属性 request.put("r1", "r1");

      4.在JSP页面使用s:property标签,使用Web元素  如:<s:property value="#request.r1"/>  

      或 <s:property value="#attr.r1"/>(不常用,虽然也可以查出,但是不精确)

      

    注:

      1.struts相当于一个中介,请求时会将一些Web元素转换为Map对象,响应时又会将这些Map转换成Web元素供jsp调用显示,

      对象通过ActionContext得到的都是 Map对象,如 Map request = (Map)ActionContext.getContext().get("request"); 所以为request添加属性,是使用put方法

      2.struts中的request等在 'Stack Context' 中,<s:debug></s:debug> 可以看到,

      在 'Stack Context'中的这些项目需要使用 '#'  来调用,<s:property value="#request.r1"/>

    访问Web元素的方法(二)(常用)

      1.在Action中,让Action实现RequestAware,SessionAware,ApplicationAware 这种接口,

      2.实现这些接口,需要重写一些方法,如setRequest(Map<String, Object> request) 这个方法

      体现了 (DI)依赖注入 和 (IOC)控制反转 

      所谓依赖注入:Action中创建的request对象需要等待这个方法的调用来对其完成注入,否则这个对象就为空了 ;

       所谓控制反转,就是控制权由应用代码中转到了外部容器,这个对象的控制权由struts容器决定

     //struts容器会检测是否实现了相应的接口,如果实现了,struts容器的调用就会调用这些方法复制,

     //所以说,这些对象的控制权交到了struts容器的手上

    访问Web元素的方法(三、四)(真实类型 HttpServletRequest, HttpSession, ServletContext)

      1.(三)在Action中,让Action实现ServletReuqestAware接口,实现该接口提供的方法后(依赖注入和控制反转),就可以得到HttpServletRequest

      通过request得到session,通过session得到ServletContext

      2.(四)在Action中,直接创建HttpServletRequest,HttpSession,ServletContext,使用ServletActionContext,得到HttpSession ServletContext

  • 相关阅读:
    BZOJ1229 USACO2008 Nov toy 玩具 【三分+贪心】*
    BZOJ1304 CQOI2009 叶子的染色 【树形DP】
    BZOJ1131 POI2008 Sta 【树形DP】
    BZOJ1096 ZJOI2007 仓库建设 【斜率优化DP】
    BZOJ4540 Hnoi2016 序列 【莫队+RMQ+单调栈预处理】*
    Codeforces 1012C Hills【DP】*
    POJ1741 Tree + BZOJ1468 Tree 【点分治】
    BZOJ2152 聪聪可可 【点分治】
    HDU1693 Eat the Trees 【插头DP】*
    RUAL1519 Formula 1 【插头DP】
  • 原文地址:https://www.cnblogs.com/xuzekun/p/7374905.html
Copyright © 2020-2023  润新知