框架的概念
1、什么是Struts2的框架
Struts2是Struts的下一代产品,是在struts1和WebWork的的技术基础上进行了合并的全新的Struts2框架
其全新的Struts2的体系结构与Struts2结构差别巨大
Struts2以WebWork为核心,采用拦截器的机制来处理用户的请求,这样的设计也使得业务逻辑控制器能够与ServletAPI完全脱离开,所以Struts2可以理解为WebWork的更新产品
虽然从Struts1到Struts2有太大的变化,但是相对于WebWork,Struts2的变化很小
2、Struts2是一个基于MVC设计模式的Web层框架
MVC和JavaEE的三层结构
MVC设计模式,是由一些网站的开发人员提出来的
JavaEE三层结构:SUN公司为EE开发划分的结构
3、常见的Web层的框架
Struts1
Struts2
WebWork
SpringMVC
4、Web层框架的特点
都是一个特点,前端控制器模式
记住:前端控制器(核心的控制器)
Struts2开启前端的控制器就是过滤器
环境准备
1、创建WEB项目,编写JSP的页面,编写超链接,点击超链接发送请求,请求服务器,让服务器的方法执行
2、下载Struts2的开发包
http://struts/apache/org 官网
3、解压struts-2.x.x-all.zip
解压后会有包和一些文件
apps --Struts2框架提供一些应用
libs --Struts2框架开发的jar包
docs --Struts2框架开发文档
src --Struts2框架源码
4、引入需要开发的jar包
Struts2框架的开发jar包非常多,但是不是所有都必须要引入,有一些必须要导入的jar包,这些jar包可以在Struts2框架提供的应用中找到
apps-> struts2-blank[空的模版].war war和zip的压缩格式是一样的,所以可以修改后缀名后解压
打开应用后 WEB-INF/lib目录下的所有jar包
5、配置Sturts2的前端控制器。必须要做,是Struts2核心的控制器
Struts2的前端控制器就是一个过滤器,需要在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>
Action类是动作类,是Struts2处理请求,封装数据,响应页面的核心控制器,需要自己写
Struts2配置文件
1、配置文件名称是struts.xml(必须是)
2、在src下引入struts.xml配置文件(路径必须在src的目录下)
Struts2执行的流程
1、执行的流程
编写页面,点击超链接,请求提交到服务器端
请求会先经过Struts2的核心过滤器(StrutsPrepareAndExecuteFilter)
过滤器的功能是完成了一部分代码功能
一系列的拦截器执行了,进行了处理工作
可以在struts-default.xml配置文件中看到很多拦截器,可以通过断点看
拦截器执行完成后,会根据struts.xml的配置文件找到请求路径,找到具体的类,通过反射的方式让方法执行
2、总结
JSP页面->StrutsPrepareAndExecuteFilter过滤器->执行一系列拦截器->执行到目标Action->返回字符串->结果页面(result)->页面跳转
struts.xml配置文件的提示
1、struts.xml配置文件 复制http://struts.apache.org/dtds/struts-2.3.dtd 没有双引号
2、window->首选项->搜索xml->选择xml Catalog
3、选择添加按钮
key type 要选择URI
key的位置把刚才复制的路径拷贝进去,http://struts.apache.org/dtds/struts-2.3.dtd
Location要在本地能找到struts2-2.3.dtd的文件,点击File System 找到
4、如果想查看源代码
选择struts-2.3.24-all.zip
框架配置文件加载的顺序
1、Struts2框架的核心是StrutsPrepareAndExecuteFilter过滤器,该过滤器有2个功能
Prepare 预处理,加载核心的配置文件
Execute 执行,让部分拦截器执行
2、StrutsPrepareAndExecuteFilter过滤器会加载什么配置文件
通过源代码可以看到具体加载的配置文件和加载配置文件的顺序
init_DefaultProperties(); 加载org/apache/struts/default.properties
init_TraditionalXmlConfiguration() 加载struts-default.xml,struts-plugin.xml,struts.xml
init_LegenctStrutsProperties(); 加载自定义的struts.propertis
init_CustomConfigurationProviders(); 加载用户自定义配置提供者
init_FilterInitParameters(); 加载web.xml
3、重点配置文件
default.properties 在org/apache/struts2/目录下,代表的是配置的是的Struts2的常量的值
struts-default.xml 在Struts2的核心包下,代表的是Struts2核心功能的配置(Bean、拦截器、结果类型等)
struts.xml 重点中的重点配置,代表WEB应用的默认配置,在工作中,基本就配置这个(可以配置常量)
web.xml 配置前端控制器(可以配置常量)
前3个配置文件是struts2框架的默认配置,基本不用修改
后3个配置文件可以允许自己修改struts2的常量,但是有个特点:后加载的配置文件修改的常量的值,会覆盖掉前面修改的常量的值
4、总结
先加载default.propertis文件,在org/apache/struts2/default.properties文件,都是常量
又加载struts-default.xml配置文件,在核心的jar包最下方,struts2框架的核心功能都是在该配置文件中配置的
再加载struts.xml的配置文件,在src目录下,代表用户自己配置的配置文件
最后加载web.xml的配置文件
后加载的配置文件会覆盖掉之前加载的配置文件(这些配置文件中可以配置常量)
5、问题
哪些配置文件中可以配置常量
default.properties 默认值,不能修改
struts.xml 可以配置,开发中基本都是在该配置文件中配置常量
struts.default.xml 可以配置,基本不会在该配置文件中配置
web.xml 可以配置,基本不会在该配置文件中配置
后加载的配置文件会覆盖掉之前加载的配置
struts.xml文件的配置
1、<package>标签,如果要配置<Action>的标签,那么必须要先配置<package>标签,代表包的概念
包含的属性
name 包的名称,要求是唯一的,管理action配置
extends 继承,可以继承其他的包,只要继承了,那么该包就包含了其他包的功能,一般都是继承struts-default
namespace 名称空间,一般与<action>标签中的name属性共同决定访问路径(怎么来访问action)常见的配置
namespace="/" 根名称空间
namespace="/aaa" 带有名称的名称空间
abstract 抽象的,这个属性基本很少使用,值如果是true,那么编写的包是被继承的
2、<action>标签
代表配置action类,包含的属性
name 和<package>标签的namespace属性一起来决定访问路径的
class 配置Action类的全路径(默认值是ActionSupport类)
method Action类中的执行方法,如果不指定,默认值是execute
3、<result>标签 可以多个标签
action类中方法执行,返回的结果跳转的页面
name 结果页面逻辑视图名称
type 结果类型(默认值为转发,也可以设置为其他的值)
Struts2配置常量
1、可以在Struts2框架中的哪些文件中配置常量
struts.xml(必须要掌握)
<constant nemw="key" value="value"></constant>
web.xml
在StrutsPrepareAndExecuteFilter配置文件中配置初始化参数
后加载的配置文件的常量会覆盖掉之前加载的常量
2、大概的了解的常量
struts.i18n.encoding=UTF-8 指定默认编码集,作用于HttpServletRequest的setCharacterEncoding方法
struts.action.extension=action 该属性指定需要Struts2处理的请求后缀,该属性的默认值是action,即所有匹配*.action的请求都由Struts2处理,
如果用户需要指定多个请求后缀,则多个后缀之间以英文逗号(,)隔开 ,,表示没有后缀的文件
struts.serve.static.browserCache=true 设置浏览器是否缓存静态内容。默认值为true(生成环境下用),开发最好关闭
struts.configuration.xml.reload=false 当struts的配置文件修改后,系统是否自动重新加载该文件,默认值为false(生产环境下用)
struts.devMode=false 开发模式下用,可以打印出更详细的错误信息
指定多个struts配置文件
1、随着应用规模增加Action的数量也会大量增加,导致struts.xml配置文件非常臃肿
为了避免struts.xml文件过于庞大,提高可读性,可以把struts.xml文件分成多个然后在struts.xml文件中包含其他配置文件
2、在<package>标签中,使用<include>标签来引入其他的struct_xx.xml的配置文件
<struts>
<include file="struts_part1.xml"/>
<include file="struts_part2.xml"/>
</struts>
3、<include file="my/demo2/struts-part1.xml"/>
Action类的三种写法
1、三种写法
Action类就是一个POJO类
POJO(Plain Ordinary Java Object)简单的Java对象,简单说就是没有继承某某个类,没有实现接口就是POJO类
Action类可以实现Action接口
Action接口中定义了5个常量,5个常量的值对应的是5个逻辑视图跳转页面(跳转的页面需要自己来配置)
还定义了一个方法,execute方法
需要掌握5个逻辑视图的常量
success 成功
input 用户数据表单校验,如果校验失败,跳转Input视图
login 登录
error 错误
none 页面不转向
Action类可以继承ActionSupport类(使用的最多)
设置错误信息
Action的访问
1、通过<action>标签中的method属性,访问到Action中具体的方法
传统的配置方式,配置更清晰更好理解,但是扩展需要修改配置文件等
页面
<a href="${pageContext.request.contextPath }/saveCust.action">保存客户</a>
<a href="${pageContext.request.contextPath }/delCust.action">删除客户</a>
配置文件
<package name="demo2" namespace="/" extends="struts-default">
<action name="saveCust" class="my.action2.CustomerAction" method="save"/>
<action name="delCust" class="my.action2.CustomerAction" method="delete"/>
</package>
Action代码
//保存客户
public String save() {
System.out.println("保存客户.");
return NONE;
}
//删除客户
public String delete() {
System.out.println("删除客户");
return NONE;
}
2、通配符的访问方式(访问的路径和方法的名称必须要有某种练习) 通配符就是*代表任意字符
使用通配符的方式可以简化配置文件的代码缩写,而且扩展和维护比较容易
页面代码
<a href="${pageContext.request.contextPath }/linkman_save.action">保存联系人</a>
<a href="${pageContext.request.contextPath }/linkman_delete.action">删除联系人</a>
配置文件
<!-- 通配符的方式 取出第一个星的值和Action类中进行匹配-->
<action name="linkman_*" class="my.action2.LinkmanAction" method="{1}">
<result name="saveOk">/demo1/suc.jsp</result>
<result name="delOk">/demo1/suc.jsp</result>
</action>
Action代码
//保存联系人
public String save() {
System.out.println("保存联系人.");
return "saveOk";
}
//删除联系人
public String delete() {
System.out.println("删除联系人");
return "delOk";
}
在JSP页面发送请求,http://localhost:8080/day06/linkman_save.action配置文件中的linkman_*可以匹配该请求,*就相当于变成了add,method属性的值使用{1}来代替,{1}就表示的是第一个*号的位置。所以method的值就等于了add,那么就找到Action类中的add方法,那么add方法就执行了
3、动态方法访问的方式(有的开发也会用)
如果想完成动态方法访问的方式,需要开启一个常量,struts.enable.DynamicMethodInvocation=false,把值设置成true
不同的Struts2框架的版本,该常量的值不一定是true或者false,需要自己来查看,如果是false们需要自己来开启
在struts.xml中开启该常量
<constant name="struts.enable.DynamicMethodInvocation" value="true"/>
页面的代码
<a href="${pageContext.request.contextPath }/user!save.action">保存小弟</a>
<a href="${pageContext.request.contextPath }/user!delete.action">删除小弟</a>
配置文件
<action name="user" class="my.action2.UserAction" ></action>
Action的类
public String save() {
System.out.println("保存客户.");
return NONE;
}
//删除客户
public String delete() {
System.out.println("删除客户");
return NONE;
}
struts.xml
<?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>
<!-- 编写常量
<constant name="struts.action.extension" value="action,,"></constant> -->
<!-- 开启动态访问 -->
<constant name="struts.enable.DynamicMethodInvocation" value="true"/>
<!-- 包结构 -->
<!-- 引入其他的配置文件 -->
<!-- <include file="my/user/struts_user.xml"></include> -->
<package name="demo1" namespace="/" extends="struts-default">
<!-- 配置Action class不写有struts2框架的默认 -->
<action name="hello" class="my.action.HelloAction" method="hello">
<!-- 配置跳转的页面,路径的写法:在sturts2框架中,不管写转发还是重定向,都不用写项目名 内部帮你加-->
<result name="ok">/demo1/suc.jsp</result>
</action>
<!-- POJO类的方式 -->
<action name="demo1Action" class="my.action.Demo1Action"></action>
<!-- 实现Action接口的方式 -->
<action name="demo2Action" class="my.action.Demo2Action" >
<result name="success">/demo1/suc.jsp</result>
</action>
<action name="demo3Action" class="my.action.Demo3Action" >
<result name="success">/demo1/suc.jsp</result>
</action>
</package>
<!-- Action的访问,传统方式 -->
<package name="demo2" namespace="/" extends="struts-default">
<action name="saveCust" class="my.action2.CustomerAction" method="save"/>
<action name="delCust" class="my.action2.CustomerAction" method="delete"/>
<!-- 通配符的方式 取出第一个星的值和Action类中进行匹配-->
<action name="linkman_*" class="my.action2.LinkmanAction" method="{1}">
<result name="saveOk">/demo1/suc.jsp</result>
<result name="delOk">/demo1/suc.jsp</result>
</action>
<!-- 配置动态方法访问 -->
<action name="user" class="my.action2.UserAction" ></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>
/**
* POJO类:没有任何继承和实现
* @author Administrator
*
*/
public class Demo1Action {
/**
* execute默认方法
* return null;不会进行跳转
* @return
*/
public String execute() {
System.out.println("Demo1Action就是POJO类");
return null;
}
}
/**
* 实现Action的接口,Action是框架提供的接口
* @author Administrator
*
*/
public class Demo2Action implements Action {
public String execute() throws Exception {
System.out.println("Demo2实现Action接口");
//return "success" 为了规范所以用常量SUCCESS
// return SUCCESS;
//页面不跳转
return NONE;
}
}
/**
* 编写Action类继承ActionSupport类,ActionSupport类已经实现了Action和一些其他接口
* @author Administrator
*
*/
public class Demo3Action extends ActionSupport{
private static final long serialVersionUID = -8507406771832040344L;
public String execute() throws Exception {
System.out.println("Demo3Action继承了ActionSupprot类");
return NONE;
}
}
/**
* Struts2框架都使用Action类处理用户的请求
* @author Administrator
*
*/
public class HelloAction {
/**
* Action类中的方法签名有要求的,必须这么做
* public 共有的
* 必须有返回值 必须String类型
* 方法名称可以是任意的,但是不能有参数列表
* 页面的跳转[默认转发]:
* 1、return 要转发的页面"字符串"
* 2、需要在struts.xml配置文件中,配置跳转的页面
*
*/
public String hello() {
//编写代码 接受请求的参数
System.out.println("Struts2");
return "ok";
}
/**
* method方法的默认值
* @return
*/
public String execute() {
System.out.println("method方法的默认值是execute");
return null;
}
}
domain.hbm.xml
<hibernate-mapping>
<!-- 配置类和表结构的映射 -->
<class name="my.domain.User" table="sys_user">
<!-- 配置id
见到name属性,JavaBean的属性
见到column属性,是表结构的字段
-->
<id name="uid" column="uid">
<!-- 主键的生成策略 -->
<generator class="native"/>
</id>
<!-- 配其他的属性 -->
<property name="username" column="username" />
<property name="password" column="password"/>
<property name="name" column="name"/>
</class>
</hibernate-mapping>
dao
public class UserDao {
/**
* 通过用户名和密码查询数据库
* @param user
* @return
*/
public User findByNameAndPwd(User user) {
//先获取
Session session = HibernateUtils.getCurrentSession();
//使用用户名和密码进行查询
Query query = session.createQuery("from User where username = ? and password=?");
//设置参数
query.setParameter(0, user.getUsername());
query.setParameter(1, user.getPassword());
//查询
List<User> list = query.list();
if(list.size()>0) {
return list.get(0);
}
return null;
}
}
service
public class UserService {
/**
* 处理登录功能
* @param user
* @return
*/
public User login(User user) {
//使用事务
Session session = HibernateUtils.getCurrentSession();
Transaction tr =session.beginTransaction();
User existUser = null;
try {
//调用持久层,查询数据
existUser = new UserDao().findByNameAndPwd(user);
tr.commit();
} catch(Exception e) {
tr.rollback();
e.printStackTrace();
}
return existUser;
}
@Test
public void run1() {
User user = new User();
user.setUsername("zs");
user.setPassword("1231");
User existUser = this.login(user);
if(existUser!=null) {
System.out.println("成功");
}
}
}
action
/**
* 用户模块的控制器
* @author Administrator
*
*/
public class UserAction extends ActionSupport{
private static final long serialVersionUID = 2822575317652909551L;
/**
* 处理登录功能
*/
public String login() {
//还需要使用request对象
//获取reqeust对象
HttpServletRequest request = ServletActionContext.getRequest();
//获取请求参数
Map<String,String[]> map = request.getParameterMap();
User user = new User();
try {
BeanUtils.populate(user, map);
//调用业务层
User existUser = new UserService().login(user);
//判断
if(existUser == null) {
//说明,用户名或者密码错误
return LOGIN;
} else {
//存入到session中
request.getSession().setAttribute("existUser", existUser);
return SUCCESS;
}
} catch (Exception e) {
e.printStackTrace();
}
System.out.println("执行了");
return NONE;
}
public String execute() {
System.out.println("啦啦");
return NONE;
}
}