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

Struts2(一)— 入门

程序员文章站 2023-03-06 08:50:01
本文主要介绍Struts2的基本入门案例,运行流程以及相关配置文件说明等。 ......

一、概述

1、什么是Struts2

  Struts2是一个基于MVC设计模式的Web应用框架,它本质上相当于一个servlet,在MVC设计模式中,Struts2作为控制器(Controller)来建立模型与视图的数据交互。Struts 2是Struts的下一代产品,是在 struts 1和WebWork的技术基础上进行了合并的全新的Struts 2框架。其全新的Struts 2的体系结构与Struts 1的体系结构差别巨大。Struts 2以WebWork为核心,采用拦截器的机制来处理用户的请求,这样的设计也使得业务逻辑控制器能够与ServletAPI完全脱离开,所以Struts 2可以理解为WebWork的更新产品。虽然从Struts 1到Struts 2有着太大的变化,但是相对于WebWork,Struts 2的变化很小。

二、Struts2入门

1.环境下载

下载地址:http://struts.apache.org/download.cgi;

Struts2(一)— 入门

2. 导入jar

​ struts的jar比较多,可以从Struts官方提供的demo中拿到必要的jar就行. 在apps/struts2-blank项目下 

Struts2(一)— 入门

3. 编写Action类

  • 新建一个类,里面定义一个方法

  • /**
     *一,创建了一个普通的类 ,定义了一个execute()方法
     *好比我们之前:创建了一个ProductServlet,然后创建了一个doGet()方法一样
     *二, 配置Action 好比配置Servlet一样,只不过不在web.xml里面配置,自己整了一套
    *   在src目录下struts.xml的文件里面配置
    */
    public class ActionDemo {
        public void execute(){
            System.out.println("收到到了请求...");
        }
     }

4. 配置struts.xml文件

  • 在src底下新建一个xml 名称为 struts.xml. 在struts.xml里面配置action

    <?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="test" extends="struts-default" namespace="/">
            <!--一个请求对应一个Servlet, struts2里面一个请求配置一个Action  -->
            <action name="demo" class="com.itheima.web.ActionDemo"></action>
        </package>
    </struts>

注意:

  1. struts.xml文件名不可随意取,必须叫做struts.xml

  2. struts.xml必须放在src类路径下

  3. 到struts的核心包中可以找到struts-2.3.dtd文件(建议配置本地的dtd,没网情况下也可以使用...)

5. 前端控制器配置

  • 在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>

    其实就是在web.xml中配置struts2的filter

6. 编写访问路径,进行测试

http://localhost:8080/day42A_Struts2/demo

三、Struts2的运行流程

1.服务器启动(项目部署)

  • 当项目部署的时候,会执行StrutsPrepareAndExecuteFilter的init()方法,在init方法里面有这样的一行代码 :

    dispatcher = init.initDispatcher(config);

  init_DefaultProperties(); // [1] —> 加载 default.properties 常量配置(国际化编码, 文件上传的size...)
  init_TraditionalXmlConfigurations(); // [2] ->加载 struts-default.xml,struts-plugin.xml,struts.xml
  init_LegacyStrutsProperties(); // [3] —> 加载struts.properties
  init_CustomConfigurationProviders(); // [5] ---> 加载自定义的一些初始化类. 一般不写
  init_FilterInitParameters() ; // [6]  ---> 加载初始化参数。 initparam
  init_AliasStandardObjects() ; // [7] ---> 给对象起别名

 

  • 图示

    顺序 配置文件名 所在位置 说明
    1 default.properties ..src\core\src\main\resources\org\apache\struts2 不能修改
    2 struts-default.xml ..\struts-2.3.32\src\core\src\main\resources 不能修改
    3 strtuts-plugin.xml 在struts2提供的插件jar包中 不能修改
    4 struts.xml 我们的应用中 我们修改的:推荐
    5 struts.properties 我们的应用中 我们修改的:不建议用
  • 后面配置的会把前面的覆盖

Struts2(一)— 入门

Struts2(一)— 入门 

2.请求到来

​ 当我们在浏览器输入请求路径 http://localhost:8080/day01A_Struts2/demo01

Struts2(一)— 入门

Struts2(一)— 入门

四、Struts.xml中的配置详解

  • eg

    <struts>
        <!--一, package指的是一组请求的集合; 一般一个模块创建一个package
                1.1 name: 包名; 随便取, 不要重复就行
                1.2 extends: 继承的意思, 我们当前的test包继承了名字叫struts-default这个包, struts-default这个包里面的功能我们test包都可以使用的
                1.3 namespace: 名称空间; 用意在于访问action的时候加一层路径(方便自己看的)
                    eg: 当前配置的是/, http://localhost:8080/day42A_Struts2/demo01
                        当前配置的是/test, http://localhost:8080/day42A_Struts2/test/demo01
                1.4 abstract: 抽象, 用意在于表明这个包是抽象的, 也就是说让别的包继承的; 如果这个包让别的包继承,习惯设置为true
           -->
        <package name="test" extends="struts-default" namespace="/" >
            <!--二, 一个请求配置一个Action标签, 不是说一个请求就要创建一个Action类
                2.1 name: action的名字,  说白了就是当前action的访问路径. 名字随便取, 不要重复就行了
                2.2 class:当前Action类的全限定名
                2.3 method: 处理当前请求的方法; 说白了也就是当前Action类里面的方法名(默认是execute)
             -->
            <action name="demo01" class="com.itheima.web.ActionDemo01"></action>
            <action name="demo01_regist" class="com.itheima.web.ActionDemo01" method="regist">
                <!--三, 配置的就是结果
                    3.1 name: 结果视图的名字; 说白了就是处理这个请求对应方法的返回值
                    3.2 type: 配置跳转的类型(默认就是转发到页面) 
                    3.3 result标签里面值: 就是要跳转的路径
                  -->
                <result name="registSuccess">
                    /registSuccess.jsp
                </result>
            </action>
        </package>
    </struts>

五、Action进阶

1、Action规范和特点

1.1 Action的编写规范
  1. action类需要提供无参构造函数

  2. action中方法规定

    ​ 修饰符:方法的修饰符必须是public

    ​ 参数列表:方法必须是无参方法

    ​ 返回值类型:String

1.2 action多例性

​   每次url访问时,action的实例就会被创建一次。action类是多实例的。

2.Action的通用写法

2.1普通类

​ 这种方式就是我们最开始给大家写的, 就是一个普通类,然后里面写一个方法,具有String类型返回值即可。

2.2实现Action接口
  • 好处是 : 我们少写一点代码, 可以使用接口里面定义的常量 SUCCESS \ ERROR ....

    public class HelloWorld02 implements Action{
        @Override
        public String execute() throws Exception {
            System.out.println("hi struts");
            return SUCCESS;
        }
    }
2.3继承ActionSupport【重点】

​   这种做法,相比较前面的好处在于, ActionSupport 虽然是实现了Action的接口 , 但是内部自己也扩展了些功能 :eg: 提供了信息的校验、并且能够根据校验的结果回到原来的页面。如 : 它里面也集成了获取国际化资源的方法,我们可以直接使用。

  • Java代码

    public class ActionDemo03 extends ActionSupport {
        @Override
        public String execute() throws Exception {
            System.out.println("ActionDemo03 execute()...");
            return SUCCESS;
        }
    }

3.Action访问的路径配置

3.1通过method属性访问【重点】

​   一般来讲,我们的action类都不会只有一个execute方法,如果存在很多的方法 ,我们如何在struts.xml 里面映射到方法里面去呢? struts提供的第一种方法是在action里面使用method属性来指定访问的具体的方法。 举例如下:

<action name="user_login" class="com.itheima.web.UserAction" method="login"></action>
<action name="user_regist" class="com.itheima.web.UserAction" method="regist"></action>
<action name="user_active" class="com.itheima.web.UserAction" method="active"></action>
<action name="user_loginout" class="com.itheima.web.UserAction" method="logout"></action>

  这样看上去是挺好的。 直接指定method ,就会找到具体的方法。 但是如果以后我们的方法很多,这样就要配置很多的action标签了。这就不太好维护了。代码先就显得很多,所以这种方式用的不多.

3.2通过通配符访问【重点】

​      通配符的访问其实还是要依赖于method的属性,只不过在匹配来访的地址 和 action的名称的时候使用通配符来匹配。采取method属性访问的方式,一个请求需要写一个Action。

  如果采取通配符的方式,只需要配置一个Action就可以了, *用于表示匹配任意字符。 后面的{1} 就表示能够取到 * 所处位置的字符,然后找到对应的方法。

<action name="user_*" class="com.itheima.web.UserAction"  method="{1}"></action>

  我们在开发中通常采取通配符方式访问.

3.3通过动态方法访问【了解】

​   第三种方式可读性不强 ,这种方式使用动态代理访问。用的很少. 使用步骤:

  1. 打开动态访问开关,在Struts.xml文件配置

    <constant name="struts.enable.DynamicMethodInvocation" value="true"></constant>
  2. 配置Action的名字

    <action name="userAction" class="com.itheima.web.UserAction"></action>
  3. 访问路径通过actionName + !+ action类中的方法名访问

    http://localhost:8080/day02D_Struts2/userAction!login

知识点补充:Constant用来配置常量值的,目的是修改Struts的default.propertis中的默认值的;

​  在Struts.xml中使用 <constant name="" value=""></constant>标签

  • 常量举例

    常量名 常量值 说明
    struts.i18n.encoding UTF-8 应用中使用的编码
    struts.multipart.maxSize 2097152 文件上传总文件大小限制:2M
    struts.action.extension action,, 能进入Struts2框架内部的url地址后缀名。多个值用逗号分隔
    struts.enable.DynamicMethodInvocation false 是否允许动态方法调用
    struts.devMode false 是否是开发模式。开发模式:改了配置文件,不需要重启。输出更多的错误信息。开发阶段建议为true。
    struts.ui.theme xhtml 页面展示用的主题
  • eg,在struts.xml里面配置常量

    <struts>
        <!-- 一,配置常量的 -->
        <!-- 1.1 打开允许动态方法访问的权限 -->
        <!-- <constant name="struts.enable.DynamicMethodInvocation" value="true"></constant> -->
        <!--1.2 添加action的访问的后缀(有用的,权限的框架(Shiro,SpringSecurity))  -->
        <constant name="struts.action.extension" value="pri,do,action,,"></constant>
        <!-- 1.3配置改了struts.xml配置文件,不需要重启服务器 -->
        <constant name="struts.devMode" value="true"></constant>
    </struts>

六、Struts2 + Hibernate整合

1. 案例需求

  • 使用Struts2 + Hibernate完成展示商品的案例.

2. 案例实现

2.1 创建web层(struts2)
  • 创建web项目, 导入jar包

  • 创建CategoryAction

    public class CategoryAction extends ActionSupport {
        
        public String findAll(){
            try {
                //1. 获得请求参数
                //2. 调用业务
                CategoryService categoryService = new CategoryService();
                List<Category> list =  categoryService.findAll();
                //3. 把list存到域里面, 转发页面 
                HttpServletRequest request = ServletActionContext.getRequest();
                request.setAttribute("list", list);
                return "findAllSuccess";
            } catch (Exception e) {
                e.printStackTrace();
                ServletActionContext.getRequest().setAttribute("msg", "查询失败...");
                return "findAllError";
            }
        }
    }
  • 在classpath(src)目录下创建struts.xml配置CategoryAction

    <?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="category" extends="struts-default" namespace="/">
            <action name="category_*" class="com.itheima.web.CategoryAction" method="{1}">
                <result name="findAllSuccess">
                    /list.jsp
                </result>
                <result name="findAllError">
                    /msg.jsp
                </result>
            
            </action>
        </package>
    </struts>
  • 在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>

3. 发现获得商品的时候出现了bug

org.hibernate.LazyInitializationException: failed to lazily initialize a collection of role: com.itheima.bean.Category.products, could not initialize proxy - no Session

  1. 使用openSession(), 不手动调用close() [千万不要用]

  2. 不使用懒加载, 再查询类别的同时就把商品查询出来

    <set name="products" fetch="select" lazy="false">
         <!--1.2 column: 外键的列名  -->
         <key column="cid"/>
         <!--1.3  class: 对方类的全限定名  -->
         <one-to-many class="com.itheima.bean.Product"/>
    </set>
  3. 还使用懒加载, 在CategoryDao使用一个商品就行了