• Struts 2.3.24源码解析+Struts2拦截参数,处理请求,返回到前台过程详析


    Struts2官网:http://struts.apache.org/

    目前最新版本:Struts 2.3.24

    Struts1已经完全被淘汰了,而Struts2是借鉴了webwork的设计理念而设计的基于MVC的框架。感兴趣的可以了解一下webwork概念,这里不做涉及。我们常说的Struts2 Spring Hibernate三大面试中常被问到的框架,其实Struts2和SpringMVC是两个独立的MVC框架。而且SpringMVC是目前最好的MVC框架(个人感觉),但是由于旧的一些项目运维需要Struts2框架的知识,故而做整理。

    下载了Struts2框架之后,我们要学习什么?1.学习框架运作的整个流程;2.学习如何搭建框架;3.自己动手写一个。先来看看框架的流程,如下:

    Struts2(由于Struts1表现层单一,无法跟Freemarker等技术整合),它采用拦截器的机制来处理用户的请求。

    先来讲讲Struts2的原理图,如上图所示:

      1.当用户发起请求时(一个URL),服务器端的Web容器收到了请求。

      2.这时,Struts2的核心控制器FilterDispatcher接受用户发起的请求,然后判断这个请求是交给action处理?还是交给web组件来处理?如果请求的action或web组件不存在,则报404错误。在整个处理过程中,需要一个辅助对象:Action映射器(ActionMapper),ActionMapper会确定调用哪个Action(这个过程的实现是依靠ActionMapper返回一个收集Action详细信息的ActionMaping对象)

      3.然后,来交给Action来处理,它会根据struts.xml的配置信息(首先执行拦截此action的所有拦截器,然后再执行请求的action对象<在这个处理过程中需要辅助对象:Action代理(ActionProxy);配置管理器(ConfigurationManager);ActionInvocation>,),

      4.Action执行完毕之后,返回一个结果(此结果用字符串来表示),这个结果经过拦截Action的所有拦截器之后,返回给主控制器。主控制器根据此结果从配置文件中找到真正的路径,然后将请求转发给相应的视图。

      5.由视图客户端作出响应。

    那么接下来我们讲讲搭建框架,最好自己亲自搭建一遍或者跟着旧的项目把流程走一遍:

    首先,将下载好的Struts2中的jar包拷贝到你构建的web project中,根据上图的设计流程,我们知道Struts2是通过过滤器,将所有的请求过滤,然后分发到各个action(action其实就是一个类,就是一个POJO类),然后根据返回的String字符串,查找Struts2的配置文件,找到应该返回到哪个页面,即可。具体如下:

    拷贝jar包到项目中,

    再来配置web.xml:

      <filter>
        <filter-name>struts2</filter-name>
        <filter-class>org.apache.struts2.dispatcher.ng.filter.StrutsPrepareAndExecuteFilter</filter-class>
      </filter>
    <filter-mapping>
        <filter-name>struts2</filter-name> 
        <url-pattern>/*</url-pattern>
    </filter-mapping>

    再来配置struts.xml配置文件(struts.xml要在src目录下)

    <struts>
        <package name="default" extends="struts-default" namespace="/">
        </package>
    </struts>

    再来创建Action(它就是一个POJO类)

    public class HelloAction{
      public String execute(){
            System.out.println("Hello Struts2!");
         return "success"; } }

    再在struts.xml中配置action和返回结果集:

    <struts>
        <package name="default" extends="struts-default" namespace="/">
         <action name="hello" class="com.hp.it.HelloAction">
           <result name="success">/hello.jsp</result>
         </action> </package> </struts>

    注意,这里的<action>标签中的name属性要与url路径中的过滤到的string相对应;class属性要与你写的action类相对应(注意要有包名).<result>标签(result是结果集)中的那么属性要与action中返回的string字符串相对应,返回的路径是WEB-INF/下的hello.jsp,即http://localhost:8080/projectname/hello.action

    再写前台的hello.jsp:

    <html>
        <head></head>
        <body><h1>Hello</h1></body>
    </html>

    整个过程就是这样的,那么问题来了:当有多个请求的时候,要写很多execute()方法么?excute方法是默认的哦!

    当然不是的,我们注意到,在struts.xml的配置文件中,标签<action>中除了name class 属性,还有一个method属性,很显然,这个标签可以指定我们映射到的方法。这个当然就是几个简单的扩张了。相应的struts.xml的配置文件:

        <package name="default" extends="struts-default" namespace="/">
         <action name="addUser" class="com.hp.it.UserAction" method="addUser">
           <result name="success">/WEB-INF/user/addUser.jsp</result>
         </action>
         <action name="updateUser" class="com.hp.it.UserAction" method="updateUser">
          <result name="success">/WEB-INF/user/updateUser.jsp</result>
        </action>
      </package>

    相应的UserAction中如下:

    public class UserAction{
          public String addUser(){
                System.out.println("addUser");
             return "success";
            }  
          public String updateUser(){
                System.out.println("updateUser");
             return "success";
            }  
    }

    那么问题又来了,一个方法写一个action,那么这样会导致配置文件中的action量很大。一种解决办法是可以在<package>标签的平行目录下,增加标签如下所示:通过增加<include>标签来导入其他的xml文件。

    <struts>
        <package name="default" extends="struts-default" namespace="/">
         <action name="hello" class="com.hp.it.HelloAction">
           <result name="success">/hello.jsp</result>
         </action>
        </package>
        <include file="otherStruts.xml">
    </struts>

    引入其他的xml文件,固然可以,但是依然无法解决action配置文件过多的问题,在整理Struts2提供了两种解决方案,如下:

    第一种URL:

    http://localhost:8080/projectname/User!add
    http://localhost:8080/projectname/User!update
    http://localhost:8080/projectname/User!list

    如上三个URL所示,User是Action类名,对应UserAction类:后面的!+方法名。

    第二种URL:

    http://localhost:8080/projectname/User?method:add
    http://localhost:8080/projectname/User?method:update
    http://localhost:8080/projectname/User?method:list

    如上三个URL所示,User是Action类名,对应UserAction类:后面的?+method:方法名。

    再接着,我们看看我们的struts.xml该如何来写?

    <package name="default" extends="struts-default" namespace="/">
         <action name="user" class="com.hp.it.UserAction" >
           <result name="add">/WEB-INF/user/addUser.jsp</result>
         </action>
         <action name="user" class="com.hp.it.UserAction" >
          <result name="update">/WEB-INF/user/updateUser.jsp</result>
         </action>
            <action name="user" class="com.hp.it.UserAction" >
          <result name="list">/WEB-INF/user/listUser.jsp</result>
         </action>
      </package>     

    注意了啊,这里的action属性name是url中那个user,即是对应着UserAction.方法是由调用的时候来决定的看具体使用谁。

    同样的,我们来看看在UserAction中应该这样来写:

    public class UserAction{
          public String addUser(){
                System.out.println("addUser");
             return "add";
            }  
          public String updateUser(){
                System.out.println("updateUser");
             return "update";
            } 
          public String listUser(){
                System.out.println("listUser");
             return "list";
            }    
    }    

     这个方法虽然减少了action的配置,但是增加了大量的结果集的配置。所有问题来了,有没有解决这个问题的方法呢?

    我们可以通过通配符来解决这个问题,这儿有一个核心思想:(约定优于配置),如下:

    <action name="*_*" class="com.hp.it.action.{1}Action" method="{2}">
         <result>/WEB-INF/{1}/{2}.jsp</result>
    </action>

    这里要强调一下,标签<result>默认的属性是 name="success"。约定优于配置,那么我们的约定是对于URl来说,它的格式应该如下面这样来向服务器端发出请求:

    http://localhost:8080/projectname/User_add
    http://localhost:8080/projectname/User_update
    http://localhost:8080/projectname/User_list

    上面这种对于URL的约定,直接可以使用通配符*_*来对它过滤。大大简化了配置文件的大小。(这种情况下,注意大小写字母)

    前面这些都是在说,服务器端的跳转,那么客户端的跳转怎么来实现呢?比如说,我们的User类在add完成之后,往往要跳转到它的list页面,这时候应该这样来配置:

    <action name="*_*" class="com.hp.it.action.{1}Action" method="{2}">
         <result>/WEB-INF/{1}/{2}.jsp</result>
         <result type="redirect" name="r_list">/{1}_list.action</result>        
    </action>

    大概看这个的含义就是说,当name=r_list的时候,进行重定向,并且重定向到{1}_list.action。相应的UserAction应该这么写

    public class UserAction{
          public String addUser(){
                System.out.println("addUser");
             return "r_list";
            }  
          public String updateUser(){
                System.out.println("updateUser");
             return "update";
            } 
          public String listUser(){
                System.out.println("listUser");
             return "list";
            }    
    }   

    如果按上述方法来做,是不是效果会更好呢。但是我们通常看到的URL,往往很少再其屁股后面加".action"这个后缀,其实这个是可以

    上面这个配置语句配置了对于.action的请求都进行过滤,同样也可以我们自己设定,如下:

    <constant name="struts.action.extension" value="action,do,zxg" />

    如上这种,当以.action;.do;.zxg的URL路径访问的时候,都会进入filter来过滤的。

    *************************************************************************************************************************

    接下来,我们再看看Struts2中是如何对参数传值做处理的(了解地址和类的对应关系;了解数据的通信(参数)的)。这部分是很关键的,而且一定要掌握清楚,不要跟SprigMVC相混淆。这段逻辑如果错误的话,调试代码的时候,介于前端和后端之间,断点加了也进不去,非常不好调控,所以在掌握原理的时候必须要掌握的清清楚楚的。那么接下来讲讲struts2传递参数的三种方案,分别如下:首先给你一个URL

    http://16.158.70.172:8080/wstax-admin/report/assetsTransactionsByRegionTime?startTime=2015-06-01&endTime=2015-06-10&_=1434004102399

    如上图,分析这个URL如下,前面的16.158.70.172是IP地址,相当于localhost,相当于127.0.0.1.然后是项目名称wstax-admin,然后是路径名称,然后我们看这个action name="assetsTransactionsByRegionTime"其后传过来三个参数,startTime和endTime分别是起始和截止时间,然后是后面的_1434004102399这个字符串,这是由于get请求的时候,加一个由时间随机生成的字符串,这样保证了每个url不同,这样每次就不会再去取缓存中的东西,而是去服务器上的东西,保证每次取的资源都是更新过后最新的资源。现在我们拦截了这个请求,传参的方法是在Action中,定义一个跟参数完全相同名字的变量,写getter和setter方法。如下:

    @Controller("dailyMonitoringAction")
    @Scope("prototype")
    public class DailyMonitoringAction extends BaseAction {
        private static final long serialVersionUID = -2065341145635610669L;
        @Autowired
        private IDailyMonitoringService dailyMonitoringService;
        private String startTime = null;
        private String endTime = null;
            public String getStartTime() {
            return startTime;
        }
        public void setStartTime(String startTime) {
            this.startTime = startTime;
        }
        public String getEndTime() {
            return endTime;
        }
        public void setEndTime(String endTime) {
            this.endTime = endTime;
        }
    }

    然后你看我们的Action中,

    /*SpringMVC传递参数和Struts传递参数 不同; Struts会调用setter方法来将值返回*/
        public String loadAssetsTransactionsByRegionTime() {
            lineVM = new LineChartVM();
            lineVM.setTitle("Assets Transactions By Region");
            lineVM.setyAxisName("Transactions");
            Map<String, Map<String, Double>> assetRegionMap = dailyMonitoringService
                    .loadAssetRegionTransactionTime(startTime, endTime);
            lineVM.setCategories(new ArrayList<String>(assetRegionMap.keySet()));
            Map<String, List<Double>> seriesMap = pivotingMap(assetRegionMap, 0D);
            List<ChartSerieVM> seriesList = new ArrayList<ChartSerieVM>();
            for (String key : seriesMap.keySet()) {
                ChartSerieVM chartSerieVM = new ChartSerieVM();
                chartSerieVM.setName(key);
                chartSerieVM.setData(seriesMap.get(key));
                seriesList.add(chartSerieVM);
            }
            lineVM.setSeries(seriesList);
            return SUCCESS;
        }

    你看我们的startTime和endTime是直接使用的,没有在函数的参数中写,而且定义的时候我们定义的是private String endTime = null;但是使用的时候,值就这么直接传递了进来,就是这么神奇啊。另外两种参数传递方法是ActionContext.getContext().put("startTime","2015-06-01");ActionContext.getContext.put("endTime","2015-06-10");(其中put进去的一对对的键值对)和通过Servlet的API来传值(ServletActionContext.getRequest.setAttribute("startTime","2015-06-01");ServletActionContext.getRequest.setAttribute("endTime","2015-06-10");)

    前台在展现数据时候,可以有如下几种方法:

    1.${startTime} ${endTime}直接取值.

    2.通过struts2的标签库<%@taglib prefix="s" uri="/struts-tags"%> 引入struts2jar包中的一个tags标签库,然后使用如下方式:

    <s:property value="#startTime">
    <s:property value="#endTime">

    就可以将数据展现出来。(注意这里的value中的变量名前面要加‘#’号。)

    备注:对于ServletActionContext.getRequest.setAttribute("endTime","2015-06-10");这种取值方式,在前台展示的时候需要这样来用,如下:

    <s:property value="#request.endTime">

    ************************************************************************************************************************************

     接下来我们看看Struts中最核心的知识点:

    鸣谢:

    参考博客(http://www.cnblogs.com/suxiaolei/archive/2011/10/28/2228063.html)

  • 相关阅读:
    Java中Timer的用法
    Java ThreadFactory接口用法
    ThreadPoolExecutor使用介绍
    linux命令学习-4-lsof
    备份文件-域名+日期扫描
    Trickbot展示新技巧:密码抓取器模块
    Trickbot增加的远程应用程序凭证抓取功能
    基于ATT和CK™框架的开放式方法评估网络安全产品
    调试键盘纪录类型的样本
    LordPE修复从进程dump出来的内存文件
  • 原文地址:https://www.cnblogs.com/RunForLove/p/4565577.html
Copyright © 2020-2023  润新知