https://www.zhihu.com/question/31565442

SpringMVC和Struts2的区别


Struts2处理请求是为每个请求都创建一个单独的Action类,Action类当中的Field属性参数作为输入和输出参数用IOC来依赖注入的方式,是基于类的。

而SpringMVC则采用输入Request和Reponse作为参数,返回ModelAndView的方式,是单例的模式,且是基于方法的模式。


作者:郭无心
链接:https://www.zhihu.com/question/31565442/answer/75692909
来源:知乎
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

无论是在web系统的开发中,还是在网络游戏的开发中,都会牵涉到request和response参数的处理问题,如何将request和reponse参数对应到我们的设计中,采用什么样的方式,处理情况是各种各种的。

-------------------------------------------
-------------------------------------------
SpringMVC的几大组件:
1)DispatcherServlet 根据请求的不同,将请求分发到不同的Controller处理器
2)Dispatcher 具体处理请求,返回逻辑视图
3)Handler Mapping
3)ModelAndView 返回的逻辑视图,注意,是逻辑视图,并不是真正的视图比如JSP页面,而是通过通过视图解析器VierResoler解气逻辑视图,返回真正的要发送给前端的视图
4)ViewResolver View 视图解析类
-------------------------------------------
struts2是通过filter实现的请求转发(或者说请求的初步处理)和初始化操作
springmvc是通过servlet实现的请求转发和初始化
-------------------待续
先上两张图
SpringMVC和Struts2的区别
SpringMVC和Struts2的区别

SpringMVC和Struts2的区别

==============自己之前整理的==============

Struts2中的Action并没有和任何ServletAPI耦合,这样框架更具灵活性,更易测试。


<package name="front_login"namespace="/front" extends="site-default">

<action name="go_login"class="frontLoginAction" method="branch">

                          <resultname="invite_success">/frontSystem/login/invite_success.jsp</result>

                          <resultname="i" type="redirect">/front/index.htm</result>

                          <resultname="success">/frontSystem/login/login.jsp</result>

                  </action>

 

Struts2 Action 对象为每一个请求产生一个实例,因此没有线程安全的问题,每次新建一个action实例去处理请求(spring是一个controller处理所有请求)

Struts2处理请求是为每个请求都创建一个单独的Action类,Action类当中的Field属性参数作为输入和输出参数用IOC来依赖注入的方式,是基于类的。

 

SpringMVC则采用输入RequestReponse作为参数,返回ModelAndView的方式,是单例的模式,且是基于方法的模式。单例的策略必须是线程安全或者同步的。

 

对于包front_login:指定了命名空间/front,则该包下所有的Action处理的URL应该是“命名空间/Action名”。如上名为go_login的Action,它处理的URL为:

本地地址/front/go_login

 

action name 相当于 Requestmapping , 触发了action类frontLoginAction中的branch方法,

然后,根据后台返回的result,选择进入哪个页面(三个result)。

 

extends=”struts-default”

这意味着该包是扩展了struts-default 包组件和拦截器,这是在struts-default.xml中文件中声明的,位于struts2-core.jar 文件的根目录。

 

 

拦截器链就是将拦截器按一定的顺序联结成一条链。在访问被拦截的方法或字段时,拦截器链中的拦截器就会按其之前定义的顺序被调用。

 

当Action请求到来的时候,会由系统的代理生成一个Action的代理对象,由这个代理对象调用Action的execute()或指定的方法,并在struts.xml中查找与该Action对应的拦截器。如果有对应的拦截器,就在Action的方法执行前(后)调用这些拦截器;如果没有对应的拦截器则执行Action的方法。其中系统对于拦截器的调用,是通过ActionInvocation来实现的。

 

拦截器的工作原理:

  当接收到一个httprequest ,

a)     当外部的httpservletrequest到来时

b)     初始到了servlet容器 传递给一个标准的过滤器链

c)      FilterDispatecher会去查找相应的ActionMapper,如果找到了相应的ActionMapper它将会将控制权限交给ActionProxy

d)     ActionProxy将会通过ConfigurationManager来查找配置struts.xml

      i. 下一步将会通过ActionInvocation来负责命令模式的实现(包括调用一些拦截Interceptor框架在调用action之前)

      ii. Interceptor做一些拦截或者初始的工作

e)     一旦action返回,会查找相应的Result

f)      Result类型可以是 jsp或者freeMark 等

g)     这些组件和ActionMapper一起返回给请求的url(注意拦截器的执行顺序)

h)     响应的返回是通过我们在web.xml中配置的过滤器

i)       如果ActionContextCleanUp是当前使用的,则FilterDispatecher将不会清理sreadlocalActionContext;如果ActionContextCleanUp不使用,则将会去清理sreadlocals。

 

  拦截器实现原理:

1、拦截器是基于java反射机制的,而过滤器是基于函数回调的。

2、过滤器依赖于servlet容器,而拦截器不依赖于servlet容器。

3、拦截器只能对Action请求起作用,而过滤器则可以对几乎所有请求起作用。

4、拦截器可以访问Action上下文、值栈里的对象,而过滤器不能。

5、在Action的生命周期中,拦截器可以多次调用,而过滤器只能在容器初始化时被调用一次。

 

  过滤器的作用:

(1)执行Actions

过滤器通过ActionMapper对象,来判断是否应该被映射到Action.如果mapper对象指示他应该被映射,过滤链将会被终止,然后Action被调用。这一点非常重要,如果同时使用SiteMesh filter,则SiteMesh filter应该放到该过滤器前,否则Action的输出将不会被装饰。

(2)清除ActionContext

过滤器为了确保内存溢出,会自动的清除ActionContext。这可能会存在一些问题,在和其它的框架集成时,例如SiteMesh。ActionContextCleanUp提供了怎么处理这些问题的一些信息。

 (3)维护静态内容

过滤器也会维护在Struts2中使用的一些公共的静态的内容,例如JavaScript文件,CSS文件等。搜索/struts/*范围内的请求,然后将/struts/后面的值映射到一些struts的公共包中,也可以在你的类路径中搜索。默认情况下会去查找以下包:org.apache.struts2.static.template。这样你只用请求/struts/xhtml/styles.css,XHTML UI主题默认的样式表将会被返回。同样,AJAX UI组件需要的JavaScript文件,也可以在org.apache.struts2.static包中被找到。如果你想加入其它被搜索的包,在web.xml中设置filter时,通过给"actionPackages"初始参数一个逗号隔开的包列表值来设定。



工作原理:

1、在浏览器中输入http://localhost:8080/Struts2Demo/hello,就会向服务器端(tomcat)发送一个请求

2、tomcat会解析URL,从中找到一个webApplication(可理解为即项目名)为Struts2Demo,然后就会在这个项目里面找到web.xml文件

3、在web.xml中找到filter标签,然后在filter中定义的filter-class处理URL中的hello(这其中其实还包含一个步骤,就是web.xml中<url-pattern>/*</url-pattern>会过滤掉所有地址,这样地址才会被filter-class中定义的类接收到)

4、StrutsPrepareAndExecuteFilter接收到地址之后,首先查询namespace(在struts.xml中的package标签中的namespace中得到它的值),然后将URL中namespace值(这里是/)后面的路径读取到(这里是hello)

5、继续在struts的action标签中查找是否有hello这个值的,如果有且发现action中有class属性,则会new一个class中声明的类,执行里面的一个execute()方法,该方法返回一个String字符串,返回该字符串之后才能得到result中的值,如果action中没有class属性,则默认有一个ActionSupport类,该类中也有一个execute方法,返回一个String值

6、上一步中讲到execute()方法,但是不一定非要执行execute()方法,当action标签中有method属性时,就会执行该属性定义的方法名称,然后同样会返回一个String字符串

7、根据返回的String字符串(success),在result中找到name属性值为返回的String字符串的标签(这里就是Action.jsp),如果没有找到,则会返回error页面;如果action中没有class则直接找到该action下面的result中的值(这里是/Hello.jsp),然后将请求forword到Action.jsp/Hello.jsp

8、最后jsp反馈到客户端。






相关文章:

  • 2022-02-09
  • 2022-02-09
  • 2022-02-21
猜你喜欢
  • 2021-10-18
  • 2021-07-17
  • 2021-08-11
  • 2021-04-04
  • 2021-09-24
  • 2022-02-09
  • 2022-02-09
相关资源
相似解决方案