一、概述
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入门
? struts的jar比较多,可以从Struts官方提供的demo中拿到必要的jar就行. 在apps/struts2-blank
项目下
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>
注意:
-
struts.xml文件名不可随意取,必须叫做struts.xml
-
struts.xml必须放在src类路径下
-
到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 我们的应用中 我们修改的:不建议用 -
后面配置的会把前面的覆盖
2.请求到来
? 当我们在浏览器输入请求路径 http://localhost:8080/day01A_Struts2/demo01
四、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的编写规范
-
action类需要提供无参构造函数
-
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通过动态方法访问【了解】
? 第三种方式可读性不强 ,这种方式使用动态代理访问。用的很少. 使用步骤:
-
打开动态访问开关,在Struts.xml文件配置
<constant name="struts.enable.DynamicMethodInvocation" value="true"></constant>
-
配置Action的名字
<action name="userAction" class="com.itheima.web.UserAction"></action>
-
访问路径通过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
-
使用openSession(), 不手动调用close() [千万不要用]
-
不使用懒加载, 再查询类别的同时就把商品查询出来
<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>
-
还使用懒加载, 在CategoryDao使用一个商品就行了