经过上篇博客,咱们能编写一个 SpringMVC 的入门实例,可是为何要这样写?这样写有啥好处?经过这篇博客咱们会有详细的了解。html
本篇博客源码下载连接:http://pan.baidu.com/s/1boOfxlP 密码:bxq4前端
经过入门实例,咱们大概知道 SpringMVC 的做用,那么它究竟是什么呢?java
Spring Web MVC是一种基于Java的实现了Web MVC设计模式的请求驱动类型的轻量级Web框架,即便用了MVC架构模式的思想,将web层进行职责解耦,基于请求驱动指的就是使用请求-响应模型,框架的目的就是帮助咱们简化开发,Spring Web MVC也是要简化咱们平常Web开发的。程序员
与之相反的是基于组件的、事件驱动的Web框架,如Tapestry、JSF等,在此就不介绍了。web
Spring Web MVC也是服务到工做者模式的实现,但进行可优化。前端控制器是DispatcherServlet;应用控制器其实拆为处理器映射器(Handler Mapping)进行处理器管理和视图解析器(View Resolver)进行视图管理;页面控制器/动做/处理器为Controller接口(仅包含ModelAndView handleRequest(request, response) 方法)的实现(也能够是任何的POJO类);支持本地化(Locale)解析、主题(Theme)解析及文件上传等;提供了很是灵活的数据验证、格式化和数据绑定机制;提供了强大的约定大于配置(惯例优先原则)的契约式编程支持。spring
第一步:用户发送请求到前端控制器(DispatcherServlet)。编程
第二步:前端控制器请求 HandlerMapping 查找 Handler,能够根据 xml 配置、注解进行查找。json
第三步: 处理器映射器 HandlerMapping 向前端控制器返回 Handler设计模式
第四步:前端控制器调用处理器适配器去执行 Handler架构
第五步:处理器适配器执行 Handler
第六步:Handler 执行完成后给适配器返回 ModelAndView
第七步:处理器适配器向前端控制器返回 ModelAndView
ModelAndView 是SpringMVC 框架的一个底层对象,包括 Model 和 View
第八步:前端控制器请求试图解析器去进行视图解析
根据逻辑视图名来解析真正的视图。
第九步:试图解析器向前端控制器返回 view
第十步:前端控制器进行视图渲染
就是将模型数据(在 ModelAndView 对象中)填充到 request 域
第十一步:前端控制器向用户响应结果
下面咱们对上面出现的一些组件进行解释:
一、前端控制器DispatcherServlet(不须要程序员开发)。 做用:接收请求,响应结果,至关于转发器,中央处理器。有了DispatcherServlet减小了其它组件之间的耦合度。 二、处理器映射器HandlerMapping(不须要程序员开发)。 做用:根据请求的url查找Handler。 三、处理器适配器HandlerAdapter(不须要程序员开发)。 做用:按照特定规则(HandlerAdapter要求的规则)去执行Handler。 四、处理器Handler(须要程序员开发)。 注意:编写Handler时按照HandlerAdapter的要求去作,这样适配器才能够去正确执行Handler 五、视图解析器ViewResolver(不须要程序员开发)。 做用:进行视图解析,根据逻辑视图名解析成真正的视图(view) 六、视图View(须要程序员开发jsp)。 注意:View是一个接口,实现类支持不一样的View类型(jsp、freemarker、pdf…) ps:不须要程序员开发的,须要程序员本身作一下配置便可。
能够总结出:须要咱们开发的工做只有处理器 Handler 的编写以及视图好比JSP页面的编写。可能你还对诸如前端控制器、处理器映射器等等名词不太理解,那么接下来咱们对其进行详细的介绍。
在 web.xml 文件中进行以下配置:
<?xml version="1.0" encoding="UTF-8"?> <web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://java.sun.com/xml/ns/javaee" xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd" id="WebApp_ID" version="3.0"> <display-name>SpringMVC_01</display-name> <!-- 配置前端控制器DispatcherServlet --> <servlet> <servlet-name>springmvc</servlet-name> <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class> <!--springmvc.xml 是本身建立的SpringMVC全局配置文件,用contextConfigLocation做为参数名来加载 若是不配置 contextConfigLocation,那么默认加载的是/WEB-INF/servlet名称-servlet.xml,在这里也就是 springmvc-servlet.xml --> <init-param> <param-name>contextConfigLocation</param-name> <param-value>classpath:springmvc.xml</param-value> </init-param> </servlet> <servlet-mapping> <servlet-name>springmvc</servlet-name> <!--第一种配置:*.do,还能够写*.action等等,表示以.do结尾的或者以.action结尾的URL都由前端控制器DispatcherServlet来解析 第二种配置:/,全部访问的 URL 都由DispatcherServlet来解析,可是这里最好配置静态文件不禁DispatcherServlet来解析 错误配置:/*,注意这里是不能这样配置的,应为若是这样写,最后转发到 jsp 页面的时候,仍然会由DispatcherServlet进行解析, 而这时候会找不到对应的Handler,从而报错!!! --> <url-pattern>*.do</url-pattern> </servlet-mapping> </web-app>
在 springmvc.xml 文件中配置。用来约束咱们所须要编码的 Handler类。
第一种配置:编写 Handler 时必需要实现 Controller
<!-- 配置处理器适配器,全部适配器都得实现 HandlerAdapter接口 --> <bean class="org.springframework.web.servlet.mvc.SimpleControllerHandlerAdapter" />
咱们能够查看源码:
第二种配置:编写 Handler 时必需要实现 HttpRequestHandler
<!-- 配置处理器适配器第二种方法,全部适配器都得实现 HandlerAdapter接口 ,这样配置全部Handler都得实现 HttpRequestHandler接口--> <bean class="org.springframework.web.servlet.mvc.HttpRequestHandlerAdapter" />
在 springmvc.xml 文件中配置。通俗来说,就是请求的 URL 到咱们这里所编写的 Handler 类的某个方法进行一些业务逻辑处理。
咱们在上面讲解了两个处理器适配器来约束 Handler,那么咱们就经过上面两种配置分别编写两个 Handler
第一种:实现Controller 接口
package com.ys.controller; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import org.springframework.web.servlet.ModelAndView; import org.springframework.web.servlet.mvc.Controller; public class HelloController implements Controller{ @Override public ModelAndView handleRequest(HttpServletRequest request, HttpServletResponse response) throws Exception { ModelAndView modelView = new ModelAndView(); //相似于 request.setAttribute() modelView.addObject("name","张三"); modelView.setViewName("/WEB-INF/view/index.jsp"); return modelView; } }
第二种:实现 HttpRequestHandler 接口
package com.ys.controller; import java.io.IOException; import javax.servlet.ServletException; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import org.springframework.web.HttpRequestHandler; public class HelloController2 implements HttpRequestHandler{ @Override public void handleRequest(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { request.setAttribute("name", "张三"); request.getRequestDispatcher("/WEB-INF/view/index.jsp").forward(request, response); } }
总结:一般咱们使用第一种方式来编写 Handler ,可是第二种没有返回值,咱们能够经过 response 修改相应内容,好比返回 json 数据。
response.setCharacterEncoding("utf-8"); response.setContentType("application/json;charset=utf-8"); response.getWriter().write("json字符串");
因此具体使用哪种根据实际状况来判断。
在 springmvc.xml 文件中配置。通俗来说就是请求的 URL 怎么能被 SpringMVC 识别,从而去执行咱们上一步所编写好的 Handler
第一种方法:
<!-- 配置Handler --> <bean name="/hello.do" class="com.ys.controller.HelloController2" /> <!-- 配置处理器映射器 将bean的name做为url进行查找,须要在配置Handler时指定bean name(就是url)--> <bean class="org.springframework.web.servlet.handler.BeanNameUrlHandlerMapping" />
这样配置的话,那么请求的 URL,必须为 http://localhost:8080/项目名/hello.do
第二种方法:
<!-- 配置Handler --> <bean id="hello1" class="com.ys.controller.HelloController" /> <bean id="hello2" class="com.ys.controller.HelloController" /> <!-- 第二种方法:简单URL配置处理器映射器 --> <bean class="org.springframework.web.servlet.handler.SimpleUrlHandlerMapping"> <property name="mappings"> <props> <prop key="/hello1.do">hello1</prop> <prop key="/hello2.do">hello2</prop> </props> </property> </bean>
这种配置请求的 URL能够为 http://localhost:8080/项目名/hello1.do,或者http://localhost:8080/项目名/hello2.do
总结:上面两种处理器映射器配置能够并存,前端控制器会正确的去判断 url 用哪一个 Handler 去处理。
第一种配置:
<!-- 配置视图解析器 进行jsp解析,默认使用jstl标签,classpath下得有jstl的包--> <bean class="org.springframework.web.servlet.view.InternalResourceViewResolver" />
若是这样配,那么在 Handler 中返回的必须是路径+jsp页面名称+".jsp"
第二种配置:
<!--配置视图解析器 --> <bean class="org.springframework.web.servlet.view.InternalResourceViewResolver"> <!-- 返回视图页面的前缀 --> <property name="prefix" value="/WEB-INF/view"></property> <!-- 返回页面的后缀 --> <property name="suffix" value=".jsp"></property> </bean>
若是这样配,那么在 Handler 中只须要返回在 view 文件夹下的jsp 页面名就能够了。
上面咱们讲解了各类配置,可能有人会问这么多配置,万一少配置了同样,那不就不能运行了,那咱们能不能不配置呢?答案是确定的,SpringMVC 给咱们提供了一个 DispatcherServlet.properties 文件。系统会首先加载这里面的配置,若是咱们没有配置,那么就默认使用这个文件的配置;若是咱们配置了,那么就优先使用咱们手动配置的。
在 SpringMVC 运行以前,会首先加载 DispatcherServlet.properties 文件里面的内容,那么咱们来看看这里面都是什么。
# Default implementation classes for DispatcherServlet's strategy interfaces. # Used as fallback when no matching beans are found in the DispatcherServlet context. # Not meant to be customized by application developers. org.springframework.web.servlet.LocaleResolver=org.springframework.web.servlet.i18n.AcceptHeaderLocaleResolver org.springframework.web.servlet.ThemeResolver=org.springframework.web.servlet.theme.FixedThemeResolver org.springframework.web.servlet.HandlerMapping=org.springframework.web.servlet.handler.BeanNameUrlHandlerMapping,\ org.springframework.web.servlet.mvc.annotation.DefaultAnnotationHandlerMapping org.springframework.web.servlet.HandlerAdapter=org.springframework.web.servlet.mvc.HttpRequestHandlerAdapter,\ org.springframework.web.servlet.mvc.SimpleControllerHandlerAdapter,\ org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter org.springframework.web.servlet.HandlerExceptionResolver=org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerExceptionResolver,\ org.springframework.web.servlet.mvc.annotation.ResponseStatusExceptionResolver,\ org.springframework.web.servlet.mvc.support.DefaultHandlerExceptionResolver org.springframework.web.servlet.RequestToViewNameTranslator=org.springframework.web.servlet.view.DefaultRequestToViewNameTranslator org.springframework.web.servlet.ViewResolver=org.springframework.web.servlet.view.InternalResourceViewResolver org.springframework.web.servlet.FlashMapManager=org.springframework.web.servlet.support.SessionFlashMapManager
咱们能够从上面得出,若是咱们不手动进行各类配置,那么也有会默认的
①、处理器适配器默认:org.springframework.web.servlet.mvc.HttpRequestHandlerAdapter
②、处理器映射器默认:org.springframework.web.servlet.handler.BeanNameUrlHandlerMapping
③、视图解析器默认:org.springframework.web.servlet.view.InternalResourceViewResolver