执行流程
- 客户端浏览器发送请求
- 根据web.xml的配置, StrutsPrepareAndExecuteFilter判断这是否是struts请求, 并通过询问ActionMapper决定需要调用某个Action
- 如果需要调用struts就把请求就转发给ActionProxy
- ActionProxy通过Configuration Manager询问框架的配置文件,找到需要调用的Action类, 找到要调用的action, ActionProxy创建实例并初始化
- ActionInvocation实例调用action, 经过一系列的拦截器, 执行里面的action方法
- Action中的方法调用业务逻辑
- Action执行结束后, ActionInvocation根据struts.xml中的配置找到对应的返回结果result(但也可能是转发或重定向到另外一个action)。
- 执行后续的各个拦截器
- 发送响应, 并跳转到相应的页面或其他action
解决了什么问题
- 简化基于mvc的web应用程序的开发
- 可以清楚的区分控制, 事务逻辑和页面显示, 提高视图的开发速度
- 文件上传和下载
- 控制action的单例和多例
- 使用ognl表达式可以快捷的访问值栈中的数据
- 强大的拦截器, 拦截用户请求
- 页面导航
Struts2拦截器
-
struts2拦截器使用的是AOP思想, 拦截器采用责任链模式
-
原理
- 在DefaultActionInvocation中就是通过递归完成所有的拦截调用操作
-
作用
- 使用拦截器进行控制action的访问
-
与过滤器Filter的区别
- 拦截器基于反射机制, 而过滤器是基于函数回调的
- 拦截器不依赖servlet容器, 而过滤器依赖servlet容器
- 拦截器只对执行到action的请求起作用, 而过滤器几乎可以对任何请求都起作用
- 拦截器可以访问action中上下文值栈中的对象, 而过滤器不能
- 在action的生命周期中, 拦截器可以多次调用, 而过滤器只能在servlet容器初始化的时候调用
-
自定义拦截器
-
创建拦截器
- 实现Interceptor接口
- 继承AbstractInterceptor类,重写抽象方法
- 需要继承MethodFilterInterceptor, 重写doIntercept方法(推荐)
-
注册拦截器
- 声明拦截器
- 在Struts的配置文件中用<interceptors>标签声明自定义的拦截器
- 引用拦截器
- 在Struts配置文件中引入自定义的拦截器
- 只要显示的使用了自定义的拦截器,那么Struts默认的拦截器就不再加载,所以还需引用Struts默认的拦截器
- 声明拦截器
-
-
文件上传与下载
- 实际上文件的上传和下载也是通过拦截器来实现的
- 文件上传
- 前台
- 请求方式必须是post
- 必须有file类型的input标签,且标签中的name属性就是后台接收时的参数名
- 提交时的encType必须是multipart/form-data
- 后台
- 接收文件的参数名必须是前台页面input标签中的name属性值
- 其他细节
- default.properties文件中定义了文件上传大小,可以通过配置文件限定文件上传的大小
- 前台
- 文件下载
- 两个头
- response.setContentType(String mimetype);
- response.setHeader("Content-disposition;filename=xxx");(XXX是文件名,注意中文乱码问题)
- 一个流
- 通过response获取输出流
- 两个头
值栈
- 值栈的内部结构
- root属性
- 是一个list结构, 里面存入的都是对象, 实现了栈的特点, 先进后出, 是OGNL上下文的一部分
- context属性
- 是ognl的上下文对象, 是一个map结构, 里面包含web阶段的三个域对象中的数据(请求域, 会话域, 应用域), 以及valueStack(值栈)中的数据
- root属性
- Struts中OGNL表达式和EL表达式
- EL表达式的查询顺序
- PageScope —>requestScope —>sessionScope —>application Scope
- OGNL表达式的查询顺序
- page Scope —>request Scope —>valueStack(根中) —>contextMap —>sessionScope —>application Scope
- EL表达式的查询顺序
为什么Struts2中的action必须是多例
- action在接收请求参数的时候,只能通过属性驱动或者模型驱动来接收数据,而这两种方式都要求在Controller层的action类中声明成员变量,如果action是单例,则在多线程并发访问时存在线程安全问题,因此action必须多例
网友评论