前言
众所周知,Struts2是个非常优秀的开源框架,我们能用Struts2框架进行开发,同时能快速搭建好一个Struts2框架,但我们是否能把Struts2框架的工作原理用语言表达清楚,你表达的原理不需要说出底层是怎么实现的,我估计大多数的人也听不懂。当面试官问到我们Struts2的工作原理时,我们该怎么回答这个问题呢?回答什么才能让自己说的与众不同,先看看其他朋友总结的
一、工作原理
在Struts2框架中的处理大概分为以下几个步骤
1 客户端初始化一个指向Servlet容器(例如Tomcat)的请求
2 这个请求经过一系列的过滤器(Filter)(这些过滤器中有一个叫做ActionContextCleanUp的可选过滤器,这个过滤器对于Struts2和其他框架的集成很有帮助,例如:SiteMesh Plugin)
3 接着FilterDispatcher被调用,FilterDispatcher询问ActionMapper来决定这个请是否需要调用某个Action
4 如果ActionMapper决定需要调用某个Action,FilterDispatcher把请求的处理交给ActionProxy
5 ActionProxy通过Configuration Manager询问框架的配置文件,找到需要调用的Action类
6 ActionProxy创建一个ActionInvocation的实例。
7 ActionInvocation实例使用命名模式来调用,在调用Action的过程前后,涉及到相关拦截器(Intercepter)的调用。
8 一旦Action执行完毕,ActionInvocation负责根据struts.xml中的配置找到对应的返回结果。返回结果通常是(但不总是,也可 能是另外的一个Action链)一个需要被表示的JSP或者FreeMarker的模版。在表示的过程中可以使用Struts2 框架中继承的标签。在这个过程中需要涉及到ActionMapper
二 工作流程
1、客户端浏览器发出HTTP请求.
2、根据web.xml配置,该请求被FilterDispatcher接收
3、根据struts.xml配置,找到需要调用的Action类和方法, 并通过IoC方式,将值注入给Aciton
4、Action调用业务逻辑组件处理业务逻辑,这一步包含表单验证。
5、Action执行完毕,根据struts.xml中的配置找到对应的返回结果result,并跳转到相应页面
6、返回HTTP响应到客户端浏览器
三、拦截器和过滤器的区别
1、拦截器是基于java反射机制的,而过滤器是基于函数回调的。
2、过滤器依赖于servlet容器,而拦截器不依赖于servlet容器。
3、拦截器只能对Action请求起作用,而过滤器则可以对几乎所有请求起作用。
4、拦截器可以访问Action上下文、值栈里的对象,而过滤器不能。
5、在Action的生命周期中,拦截器可以多次调用,而过滤器只能在容器初始化时被调用一次。
四、什么要使用Struts2
Struts2 是一个相当强大的Java Web开源框架,是一个基于POJO的Action的MVC Web框架。它基于当年的Webwork和XWork框架,继承其优点,同时做了相当的改进。
1、Struts2基于MVC架构,框架结构清晰,开发流程一目了然,开发人员可以很好的掌控开发的过程。
2、使用OGNL进行参数传递。
OGNL提供了在Struts2里访问各种作用域中的数据的简单方式,你可以方便的获取Request,Attribute,Application,Session,Parameters中的数据。大大简化了开发人员在获取这些数据时的代码量。
3、强大的拦截器
Struts2 的拦截器是一个Action级别的AOP,Struts2中的许多特性都是通过拦截器来实现的,例如异常处理,文件上传,验证等。拦截器是可配置与重用的,可以将一些通用的功能如:登录验证,权限验证等置于拦截器中以完成一些Java Web项目中比较通用的功能。在我实现的的一Web项目中,就是使用Struts2的拦截器来完成了系统中的权限验证功能。
4、易于测试
Struts2的Action都是简单的POJO,这样可以方便的对Struts2的Action编写测试用例,大大方便了5Java Web项目的测试。
易于扩展的插件机制在Struts2添加扩展是一件愉快而轻松的事情,只需要将所需要的Jar包放到WEB-INF/lib文件夹中,在struts.xml中作一些简单的设置就可以实现扩展。
6、模块化管理
Struts2已经把模块化作为了体系架构中的基本思想,可以通过三种方法来将应用程序模块化:将配置信息拆分成多个文件把自包含的应用模块创建为插件创建新的框架特性,即将与特定应用无关的新功能组织成插件,以添加到多个应用中去。
7、全局结果与声明式异常
为应用程序添加全局的Result,和在配置文件中对异常进行处理,这样当处理过程中出现指定异常时,可以跳转到特定页面。
他的如此之多的优点,是很多人比较的青睐,与spring ,Hibernate进行结合,组成了现在比较流行的ssh框架,当然每个公司都要自己的框架,也是ssh变异的产品。
五、看到网友的对Struts2的原理总结,我自己也总结以便后续的面试,以下是我的疑问
1、客服端发起一个请求,通过HTTP协议指向Tomcat容器,tomcat拿到请求她干了什么?
2、我们web.xml配置
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
|
<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配置文件中可以看到org.apache.struts2.dispatcher.ng.filter.StrutsPrepareAndExecuteFilter
,这个Filter这个类有什么用?
这个 /* 是拦截所有的请求,他拦截了请求做了什么处理? <url-pattern>/*</url-pattern>
3、我们struts.xml配置
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
<struts>
<package name="default" namespace="/" extends="struts-default">
<action name="hello">
<result>
/Hello.jsp
</result>
</action>
</package>
</struts>
|
谁负责根据struts.xml配置,找到需要调用的Action类和方法呢?
4、最后根据struts.xml中的配置找到对应的返回结果result,在返回HTTP响应到客户端浏览器
struts2的工作原理
- 客户端发送请求
- 经过一系列的过滤器
- FilterDispatcher通过ActionMapper来决定这个REquest需要调用的Action
- FilterDispather交给ActionProxy
- 通过ConfigurationManager询问struts.xml找到对应的Action
- ActionProxy创建一个ActionInvocation实例
- 调用Action Action执行完毕ActionInvocation负责根据struts.xml中的配置找到对应的返回结果
Struts拦截器,常用的拦截器
- Struts2的拦截器是其核心部分,它提供了一种机制,可以定义一个特殊的模块,对请求进行封装和处理,这个模块在Action执行前后都可以进行
- modelDriven(模型驱动) servletConfig(获取ServletApo) params(动态参数注入) validation(输入验证 声明式验证)
struts2的优点
- 实现了mvc模式 层次结构清晰,
- 丰富的标签库,提高了开发效率
- 提供了丰富的拦截器实现
- 方便异常处理
- 通过配置文件可以掌握系统各个部分之间的关系
struts2核心控制器
- FilterDispatcher
- 它负责拦截所有的用户请求,当 用户请求到达时,该Filter会过滤用户请求。如果用户请求以action结尾,该请求将被转入Struts 2框架处理。
struts2配置文件覆盖顺序
- struts.xml -> struts.properties -> web.xml(配置过滤器时,指定参数。程序员可以编写)
struts2修改常量的方式
-
struts.xml :
<constant name="struts.action.extension" value="do"/>
-
struts.properties :
struts.action.extension=do
Action
- Action为多例的,所以在Spring中Scope要设置参数
- 通配符 动态调用
Action实现的三种方式
- 单纯的POJO
- 实现Action接口 (提供了格式化的返回参数)
- 继承ActionSupport(实现了Action接口并且其中还做了一些国际化和验证的功能)
访问ServletAPI
- ServletActionContext
- 实现接口 ServletRequestAware,ServletResponseAware ,ServletContextAware 拦截器负责注入(ServletConfig)
结果视图
- chain(转发到动作,要是不在同一个包内需要设置 name) dispatcher(默认) redierct (请求重定向) stream(文件上传和下载)
默认的struts-default有什么作用
- struts-default包是由struts内置的,它定义了struts2内部的众多拦截器和Result类型,而Struts2很多核心的功能都是通过这些内置的拦截器实现,当包继承了struts-default包才能使用struts2为我们提供的这些功能。
- 通常每个包都应该继承struts-default包。
struts2如何对指定的方法进行验证
- 验证功能是由validation拦截器来负责处理的。回显错误信息是由workflow拦截器来负责处理的。
- 验证失败以后错误信息回显需要设置 <result name="input">/regist.jsp</result> 结果视图
- 对类中所有方法进行验证,复写validate()方法,当出错的时候addFieldError("name", "请输入用户名");
- 对指定方法:将validate()方法改写成为public void validateDemo1 后面是指定方法的名字(验证Demo1方法)
值栈
- ValueStack的生命周期为Action的生命周期,保存在request域中,
- 每个action都有与之对应的值栈,action对象默认保存在栈顶
- 使用OGNL访问值栈中的内容不需要添加# 而访问其他域中的需要添加#
ActionContext、ServletContext、pageContext的区别?
- ActionContext是当前Action的上下文环境,通过ActionContext可以获取到request、session、ServletContext等与Action有关的对象的引用
- ServletContext是域对象,一个web应用中只有一个ServletContext,生命周期伴随整个web应用;
- pageContext是JSP中的最重要的一个内置对象,可以通过pageContext获取其他域对象的应用,同时它是一个域对象,作用范围只针对当前页面
拦截器
- 每个拦截器都是实现了Interceptor接口的java类
- Init() 在这个拦截器初始化的时候被调用
- intercept() 当拦截器起作用的时候调用
- destroy() 在拦截器被销毁的时候调用
- struts2默认有18个拦截器
自定义拦截器
- 编写一个类继承 MethodFilterInterceptor
- struts2.xml中声明拦截器
- struts2.xml中声配置拦截器 , 当指定自定义拦截器时默认拦截器会失效,所以还要显示的在指定一次.
文件上传
-
file域 form表单
enctype=multipart/form-data
提交方式为post - action接受文件对象为表单中的name, 文件类型为name+ContentType 文件名字为:jsp页面file组件的名称+FileName
- 可以用数组来接收多个文件的上传
我觉的要说清楚Struts原理,不光死记硬背资料上的写的原理,还要带着这些问题去做总结。
总结
以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对服务器之家的支持。
原文链接:http://www.linuxidc.com/Linux/2017-11/148626.htm