欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

【Struts2】简介及入门

程序员文章站 2022-05-29 09:41:19
一、概述二、Struts2 快速入门程序2.1 开发流程比较2.2 引入依赖2.2 创建jsp页面2.3 在web.xml中配置前端控制器2.4 创建struts.xml配置文件2.4 创建一个HelloAction类2.5 在struts.xml文件中配置HelloAction2.6 在index... ......

一、概述

  1. 问题:什么是框架,框架有什么用?
    • 框架 是 实现部分功能的代码 (半成品),使用框架简化企业级软件开发 ,提高开发效率。
    • 学习框架 ,要清楚的知道框架能做什么,还有哪些工作需要自己编码实现
  2. 问题:什么是struts2框架,它有什么用?
    • struts2 是struts的下一代产品,是在 struts1 和 webwork 的技术基础上进行了合并的全新的struts2框架。其全新的struts2的体系结构与struts1的体系结构差别巨大。struts2以webwork为核心,struts2=struts1+webwork
    • struts2框架是apache产品。
    • struts2是一个标准的mvc框架。
    • javaweb中的 model2 模式就是一个mvc模式
    • model2=servlet+jsp+javabean
    • struts2框架是在javaweb开发中使用的。
    • 使用struts2框架,可以简化我们的web开发,并且降低程序的耦合度。
  3. 类似于struts2框架的产品 :
    • struts1 webwork jsf springmvc
    • ssh --- struts2 + spring + hibernate
    • ssm --- springmvc + spring + mbatis
  4. xwork --- 它是webwork核心
    • xwork提供了很多核心功能:前端拦截机(interceptor),运行时表单属性验证,类型转换,强大的表达式语言(ognl – the object graph navigation language),ioc(inversion of control反转控制)容器等

二、struts2 快速入门程序

2.1 开发流程比较

  • web开发流程:index.jsp------>helloservlet-------->hello.jsp
  • struts2流程:index.jsp------>helloaction--------->hello.jsp

2.2 引入依赖

<!-- https://mvnrepository.com/artifact/org.apache.struts/struts2-core -->
<dependency>
    <groupid>org.apache.struts</groupid>
    <artifactid>struts2-core</artifactid>
    <version>2.3.16.3</version>
</dependency>

2.2 创建jsp页面

  • 创建index.jsp页面(后面修改)
  • 创建hello.jsp页面(内容如下)
<%@ page language="java" import="java.util.*" pageencoding="utf-8"%>
<!doctype html public "-//w3c//dtd html 4.01 transitional//en">
<html>
  <head> 
    <title>my jsp 'index.jsp' starting page</title>
  </head>
  
  <body>
    <h1>hello struts2</h1>
  </body>
</html>

2.3 在web.xml中配置前端控制器

  • 在web.xml文件中配置前端控制器(核心控制器)-----就是一个filter
<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>

2.4 创建struts.xml配置文件

  • 在src下(classes下)创建一个struts.xml配置文件 ,这个是struts2框架配置文件。

2.4 创建一个helloaction类

  • 要求,在helloaction类中创建一个返回值是string类型的方法,注意,无参数。
public class helloaction {
    
    public string say() {
        return "good";
    }
}

2.5 在struts.xml文件中配置helloaction

<?xml version="1.0" encoding="utf-8" ?>
<!doctype struts public
    "-//apache software foundation//dtd struts configuration 2.3//en"
    "http://struts.apache.org/dtds/struts-2.3.dtd">

<struts>

    <package name="default" namespace="/" extends="struts-default">
        <action name="hello" class="com.hao.action.helloaction"
            method="say">
            <result name="good">/hello.jsp</result>
        </action>
    </package>
    
</struts>

2.6 在index.jsp中添加连接,测试

<%@ page language="java" import="java.util.*" pageencoding="utf-8"%>
<!doctype html public "-//w3c//dtd html 4.01 transitional//en">
<html>
  <head>  
    <title>my jsp 'index.jsp' starting page</title>
  </head>
  <body>
    <a href="${pagecontext.request.contextpath}/hello">第一次使用struts2</a>
  </body>
</html>

2.7 测试

  • 在地址栏中输入:http://localhost:8080/struts2-001-entryp/index.jsp 访问连接,就可以看到 helloaction类中的say方法执行了,也跳转到了hello.jsp.

三、对入门程序进行流程分析

  • 模仿struts2流程完成入门程序:

3.1 创建项目,引入maven依赖

<dependency>
    <groupid>org.apache.struts</groupid>
    <artifactid>struts2-core</artifactid>
    <version>2.3.16.3</version>
</dependency>

<dependency>
    <groupid>javax.servlet</groupid>
    <artifactid>servlet-api</artifactid>
    <version>3.0-alpha-1</version>
    <scope>provided</scope>
</dependency>

<!-- https://mvnrepository.com/artifact/dom4j/dom4j -->
<dependency>
    <groupid>dom4j</groupid>
    <artifactid>dom4j</artifactid>
    <version>1.6.1</version>
</dependency>
<!-- https://mvnrepository.com/artifact/commons-beanutils/commons-beanutils -->
<dependency>
    <groupid>commons-beanutils</groupid>
    <artifactid>commons-beanutils</artifactid>
    <version>1.9.3</version>
</dependency>

3.2 自定义strusfilter 过滤器

  • 1.创建一个filter----strutsfilter
  • 2.在web.xml文件中配置strutsfilter
<!doctype web-app public
 "-//sun microsystems, inc.//dtd web application 2.3//en"
 "http://java.sun.com/dtd/web-app_2_3.dtd" >

<web-app>
  <display-name>archetype created web application</display-name>
  <filter>
    <filter-name>strutsfilter</filter-name>
    <display-name>strutsfilter</display-name>
    <description></description>
    <filter-class>com.hao.filter.strutsfilter</filter-class>
  </filter>
  <filter-mapping>
    <filter-name>strutsfilter</filter-name>
    <url-pattern>/*</url-pattern>
  </filter-mapping>
</web-app>
  • 3.在strutsfilter中完成拦截操作,并访问action中的方法,跳转到hello.jsp页面操作.
public class strutsfilter implements filter {

    public void init(filterconfig filterconfig) throws servletexception {

    }

    public void dofilter(servletrequest req, servletresponse resp, filterchain chain)
            throws ioexception, servletexception {

        // 1.强转
        httpservletrequest request = (httpservletrequest) req;
        httpservletresponse response = (httpservletresponse) resp;

        // 2.操作

        // 2.1 得到请求资源路径
        string uri = request.getrequesturi();
        string contextpath = request.getcontextpath();
        string path = uri.substring(contextpath.length() + 1);

        // system.out.println(path); // hello

        // 2.2 使用path去struts.xml文件中查找某一个<action name=path>这个标签
        saxreader reader = new saxreader();

        try {
            // 得到struts.xml文件的document对象。
            document document = reader.read(new file(this.getclass().getresource("/struts.xml").getpath()));

            element actionelement = (element) document.selectsinglenode("//action[@name='" + path + "']"); // 查找<action
                                                                                                            // name='hello'>这样的标签

            if (actionelement != null) {
                // 得到<action>标签上的class属性以及method属性
                string classname = actionelement.attributevalue("class"); // 得到了action类的名称
                string methodname = actionelement.attributevalue("method");// 得到action类中的方法名称。

                // 2.3通过反射,得到class字节码对象,得到method对象
                class<?> actionclass = class.forname(classname);
                method method = actionclass.getdeclaredmethod(methodname);

                // 处理请求参数封装:

                object actionobj = actionclass.newinstance();

                // 2.模型驱动
                if (actionobj instanceof mymodeldriven) {
                    mymodeldriven mmd = (mymodeldriven) actionobj;

                    beanutils.populate(mmd.getmodel(), request.getparametermap());
                } else {
                    // 1.属性驱动
                    beanutils.populate(actionobj, request.getparametermap());//
                }

                // 2.4 让method执行.
                string returnvalue = (string) method.invoke(actionobj); // 是让action类中的方法执行,并获取方法的返回值。

                // 2.5
                // 使用returnvalue去action下查找其子元素result的name属性值,与returnvalue做对比。
                element resultelement = actionelement.element("result");
                string namevalue = resultelement.attributevalue("name");

                if (returnvalue.equals(namevalue)) {
                    // 2.6得到了要跳转的路径。
                    string skippath = resultelement.gettext();

                    // system.out.println(skippath);

                    request.getrequestdispatcher(skippath).forward(request, response);
                    return;
                }
            }

        } catch (documentexception e) {
            e.printstacktrace();
        } catch (classnotfoundexception e) {
            e.printstacktrace();
        } catch (nosuchmethodexception e) {
            e.printstacktrace();
        } catch (securityexception e) {
            e.printstacktrace();
        } catch (illegalaccessexception e) {
            e.printstacktrace();
        } catch (illegalargumentexception e) {
            e.printstacktrace();
        } catch (invocationtargetexception e) {
            e.printstacktrace();
        } catch (instantiationexception e) {
            e.printstacktrace();
        }

        // 3.放行
        chain.dofilter(request, response);

    }

    public void destroy() {
    }

}

3.3 流程分析

  • 请求 ---- strutsprepareandexecutefilter 核心控制器 ----- interceptors 拦截器(实现代码功能 ) ----- action 的execute --- 结果页面 result

四、struts2配置

4.1 struts2 配置文件加载顺序

  • struts2框架要能执行,必须先加载strutsprepareandexecutefilter.
  • 在strutsprepareandexecutefilter的init方法中对dispatcher进行了初始化.
  • 在dispatcher类中定义的init方法内就描述了struts2配置文件加载的顺序
// [1]   org/apache/struts2/default.properties 
init_defaultproperties();
// [2]  struts-default.xml,struts-plugin.xml,struts.xml 
init_traditionalxmlconfigurations(); 
// [3] --- 自定义struts.properties  (源码中的注释没有[4])
init_legacystrutsproperties(); 
// [5]  ----- 自定义配置提供
init_customconfigurationproviders(); 
// [6] ----- web.xml 
init_filterinitparameters() ; 
// [7] ---- bean加载 
init_aliasstandardobjects() ; 
1.default.properties文件
    作用:定义了struts2框架中所有常量
    位置: org/apache/struts2/default.properties ,struts2-core.jar包下
    
2.struts-default.xml
    作用:配置了bean,interceptor,result等。
    位置:在struts的core核心jar包.
    
  struts-plugin.xml
    它是struts2框架中所使用的插件的配置文件。
  struts.xml              
    我们使struts2所使用的配置文件。
        
3.自定义的struts.properties
    我们可以自定义常量。
    
4.web.xml
    
在开发中,后加载文件中的配置会将先加载文件中的配置覆盖。我们一般要记住如下顺序即可:
default.properties
struts-default.xml
struts.xml

4.2 关于action的配置

  1. <package> 作用:是用于声明一个包。用于管理action。它的常用属性如下
    • name: 它用于声明一个包名,包名不能重复,也就是它是唯一的。
    • namespace: 它与action标签的name属性合并确定了一个唯一访问action的路径。
    • extends: 它代表继承的包名。
    • abstrace: 它可以取值为true/false,如果为true,代表这个包是用于被继承的。
  2. <action> 用于声明 一个action,它的常用属性如下:
    • name: 就是action的一个名称,它是唯一的(在同包内) 它与package中的namespace确定了访问action的路径。
    • class: action类的全名
    • method: 要访问的action类中的方法的名称,方法无参数 ,返回值为string.
  3. <result> 用于确定返回结果类型,它的常用属性如下:
    • name 它与action中的方法返回值做对比,确定跳转路径。

4.3 关于action配置其它细节:

  1. 关于默认值问题
    • <package namespace="默认值"> -- namespace的默认值是 ""
    • <action class="默认值" method="默认值"> -- class的默认值是 "com.opensymphony.xwork2.actionsupport" ,method的默认值是execute
    • <result name="默认值"> name的默认值是 "success"
  2. 关于访问action的路径问题 ,现在的action的配置是:

    <package name="default" namespace="/" extends="struts-default">
    <action name="hello" class="com.hao.action.defaultaction">
    <result>/hello.jsp</result>
    </action>
    </package>
    • 当我们输入: http://localhost/struts2-003-exerconfig/a/b/c/hello, 也可以访问到了action。
    • 原因:struts2中的action被访问时,它会首先查找
    • txt 1.namespace="/a/b/c" action的name=hello 没有.
      2.namespace="/a/b action的name=hello 没有
      3.namespace="/a" action的name=hello 没有
      4.namespace="/" action的name=hello 查找到了.

    • 如果最后也查找不到,会报404错误.
  3. 默认的action。

    • 作用:处理其它action处理不了的路径。
    • <default-action-ref name="action的名称" /> ,配置了这个,当访问的路径,其它的action处理不了时,就会执行name指定的名称的action。
  4. action的默认处理类

    • 在action配置时,如果class不写。默认情况下是com.opensymphony.xwork2.actionsupport
    • <default-class-ref class="com.hao.action.defaultaction"/>,如果设置这个了,那么在当前包下,默认处理action请的的处理类就为class指定的类。

4.4 关于常量配置

  • default.properties 它声明了struts中的常量。
  • 问题:人为设置常量,可以在哪些位置设置 ?

    • 1.struts.xml(应用最多),<constant name="常量名称" value="常量值"></constant>
    • 2.struts.properties(基本不使用)
    • 3.web.xml(了解),配置常量,是使用strutsprepareandexecutefilter的初始化参数来配置的.
    <init-param>
        <param-name>struts.action.extension</param-name>
    <param-value>do,,</param-value>
    </init-param>
  • 常用常量

    • struts.action.extension=action,, -- 这个常量用于指定strus2框架默认拦截的后缀名.
    • <constant name="struts.i18n.encoding" value="utf-8"/> -- 相当于request.setcharacterencoding("utf-8"); 解决post请求乱码
    • <constant name="struts.serve.static.browsercache" value="false"/> -- false不缓存,true浏览器会缓存静态内容,生产环境设置true、开发环境设置false
    • <constant name="struts.devmode" value="true" /> , 提供详细报错页面,修改struts.xml后不需要重启服务器

3.5 struts.xml文件的分离:

  • 目的:就是为了阅读方便。可以让一个模块一个配置文件,在struts.xml文件中通过<include file="test.xml"/> 导入其它的配置文件。

五、action

5.1 action类的创建方式

  • 三种创建方式

  1. 创建一个pojo类.
    • 简单的java对象(plain old java objects)
    • 指的是没有实现任何接口,没有继承任何父类(除了object)
    • 优点:无耦合。
    • 缺点:所有工作都要自己实现。
    • 在struts2框架底层是通过反射来操作:
      • struts2框架 读取struts.xml 获得 完整action类名
      • obj = class.forname("完整类名").newinstance();
      • method m = class.forname("完整类名").getmethod("execute"); m.invoke(obj); 通过反射 执行 execute方法
  2. 创建一个类,实现action接口.(com.opensymphony.xwork2.action)

    • 优点:耦合低。提供了五种结果视图,定义了一个行为方法。
    • 缺点:所有工作都要自己实现。
    public static final string success = "success";  // 数据处理成功 (成功页面)
    public static final string none = "none";  // 页面不跳转  return null; 效果一样
    public static final string error = "error"; // 数据处理发送错误 (错误页面)
    public static final string input = "input"; // 用户输入数据有误,通常用于表单数据校验 (输入页面)
    public static final string login = "login"; // 主要权限认证 (登陆页面)
  3. 创建一个类,继承自actionsupport类. (com.opensymphony.xwork2.actionsupport)

    • actionsupport类实现了action接口。
    • 优点:表单校验、错误信息设置、读取国际化信息 三个功能都支持.
    • 缺点:耦合度高。
    • 在开发中,第三种会使用的比较多.

5.2 关于action的访问

  • 1.通过设置method的值,来确定访问action类中的哪一个方法.
    • <action name="book_add" class="com.hao.action.bookaction" method="add"></action>
    • 当访问的是book_add,这时就会调用bookaction类中的add方法。
    • <action name="book_update" class="com.hao.action.bookaction" method="update"></action>
    • 当访问的是book_update,这时就会调用bookaction类中的update方法。
  • 2.使用通配符来简化配置

    • 在struts.xml文件 中配置
    • <action name="*_*" class="com.hao.action.{1}action" method="{2}"></action>
    • jsp页面
    • book.jsp
    <a href="${pagecontext.request.contextpath}/book_add">book add</a><br>
    <a href="${pagecontext.request.contextpath}/book_update">book update</a><br>
    <a href="${pagecontext.request.contextpath}/book_delete">book delete</a><br>
    <a href="${pagecontext.request.contextpath}/book_search">book search</a><br>
    • product.jsp
    <a href="${pagecontext.request.contextpath}/product_add">product add</a><br>
    <a href="${pagecontext.request.contextpath}/product_update">product update</a><br>
    <a href="${pagecontext.request.contextpath}/product_delete">product delete</a><br>
    <a href="${pagecontext.request.contextpath}/product_search">product search</a><br>
    • 当访问book add时,这时的路径是 book_add,那么对于struts.xml文件中.
    • 第一个 * 就是 book
    • 第二个 * 就是 add
    • 对于{1}action---->bookaction
    • 对于method={2}--->method=add
    • 使用通配符来配置注意事项:
    • 1.必须定义一个统一的命名规范。
    • 2.不建议使用过多的通配符,阅读不方便。
  • 3.动态方法调用 (了解)

    • 在struts.xml文件中
    • <action name="book" class="com.hao.action.bookaction"></action> 访问时路径: http://localhost/struts2-003-exerconfig/book!add
      就访问到了bookaction类中的add方法。
    • 对于book!add 这就是动态方法调用。
    • 注意:struts2框架支持动态方法调用,是因为在default.properties配置文件中设置了,动态方法调用为true. struts.enable.dynamicmethodinvocation = true

5.3 在struts2框架中获取servlet api

  • 对于struts2框架,不建议直接使用servlet api;
  • 在struts2中获取servlet api有三种方式:

  • 1.通过actioncontext来获取

    • 获取一个actioncontext对象。 actioncontext context=actioncontext.getcontext()
    • 获取servlet api
    • 注意:通过actioncontext获取的不是真正的servlet api,而是一个map集合。
    1.context.getapplication()
    2.context.getsession()
    3.context.getparameter();---得到的就相当于request.getparametermap()
    4.context.put(string,object) 相当于request.setattribute(string,string);
  • 2.注入方式获取(这种方式是真正的获取到了servlet api)

    • 要求action类必须实现提定接口。
    • servletcontextaware :注入servletcontext对象
    • servletrequestaware :注入 request对象
    • servletresponseaware :注入response对象

    • 重写接口中的方法。

    • 声明一个web对象,使用接口中的方法的参数对声明的web对象赋值.

    //获取servlet api  通过注入方式
    public class servletdemo2action extends actionsupport implements
    servletrequestaware {
    private httpservletrequest request;
    @override
    public string execute() throws exception {
    system.out.println(request.getparameter("username"));
    return null;
    }
    public void setservletrequest(httpservletrequest request) {
    this.request = request;
    }
    }
    • 扩展:分析其实现,是使用struts2中的一个interceptor完成的.( struts-default.xml中的拦截器)
    <interceptor name="servletconfig" class="org.apache.struts2.interceptor.servletconfiginterceptor"/>
    
         if (action instanceof servletrequestaware) { //判断action是否实现了servletrequestaware接口
            httpservletrequest request = (httpservletrequest) context.get(http_request); //得到request对象.
    ((servletrequestaware) action).setservletrequest(request);//将request对象通过action中重写的方法注入。
    }
  • 3.通过servletactioncontext获取.在servletactioncontext中方法都是static。

//获取servlet api  通过servletactioncontext获取
public class servletdemo3action extends actionsupport {

    @override
    public string execute() throws exception {
        httpservletrequest request = servletactioncontext.getrequest();

        system.out.println(request.getparameter("username"));
        return success;
    }

}

5.4 result结果类型

  • 标签
  • 1.name 与action中的method的返回值匹配,进行跳转.
  • 2.type 作用:是用于定义跳转方式
  • 对于type属性它的值有以下几种:在struts-default.xml文件中定义了type可以取的值
<result-type name="chain" class="com.opensymphony.xwork2.actionchainresult"/>
<result-type name="dispatcher" class="org.apache.struts2.dispatcher.servletdispatcherresult" default="true"/>
<result-type name="freemarker" class="org.apache.struts2.views.freemarker.freemarkerresult"/>
<result-type name="httpheader" class="org.apache.struts2.dispatcher.httpheaderresult"/>
<result-type name="redirect" class="org.apache.struts2.dispatcher.servletredirectresult"/>
<result-type name="redirectaction" class="org.apache.struts2.dispatcher.servletactionredirectresult"/>
<result-type name="stream" class="org.apache.struts2.dispatcher.streamresult"/>
<result-type name="velocity" class="org.apache.struts2.dispatcher.velocityresult"/>
<result-type name="xslt" class="org.apache.struts2.views.xslt.xsltresult"/>
<result-type name="plaintext" class="org.apache.struts2.dispatcher.plaintextresult" />
  • 必会: chain dispatcher redirect redirectaction stream
  • dispatcher:它代表的是请求转发,也是默认值。它一般用于从action跳转到页面。
  • chain:它也相当于请求转发。它一般情况下用于从一个action跳转到另一个action。
  • redirect:它代表的是重定向 它一般用于从action跳转到页面
  • redirectaction: 它代表的是重定向 它一般用于从action跳转另一个action。
  • stream:代表的是服务器端返回的是一个流,一般用于下载。

  • 局部结果页面与全局结果页面

<package name="default" namespace="/" extends="struts-default">
    <!-- 全局结果页面 -->
    <global-results>
        <result>/demo1_success.jsp</result>
    </global-results>

    <action name="demo1" class="com.hao.action.servletdemo1action">
        <!-- 局部结果页面 -->
    </action>

    <action name="demo2" class="com.hao.action.servletdemo2action">
        <!-- <result>/demo1_success.jsp</result> -->
    </action>

    <action name="demo3" class="com.hao.action.servletdemo3action">
        <!-- <result type="redirect">/demo1_success.jsp</result> -->
    </action>

</package>