一、SpringMVC详细介绍

Spring Web MVC是一种基于Java的实现了Web MVC设计模式的请求驱动类型的轻量级Web框架,即使用了MVC架构模式的思想,将web层进行职责解耦,基于请求驱动指的就是使用请求-响应模型,框架的目的就是帮助我们简化开发,Spring Web MVC也是要简化日常Web开发的。

Spring Web MVC前端控制器是DispatcherServlet;应用控制器其实拆为处理器映射器( Handler Mapping)进行处理器管理和视图解析器(View Resolver)进行视图管理;页面控制器/动作/处理器为Controller接口(仅包含ModelAndView handleRequest(request, response) 方法)的实现(也可以是任何的POJO类);支持本地化(Locale)解析、主题(Theme)解析及文件上传等;提供了非常灵活的数据验证、格式化和数据绑定机制;提供了强大的约定大于配置(惯例优先原则  约定大于配置!)的契约式编程支持。

二、SpringMVC处理请求流程:(图是网上随便找的)

springmvc项目如何配置dbcp_xml

第一步:用户发送请求到前端控制器(DispatcherServlet)。

第二步:前端控制器请求 HandlerMapping 查找 Handler(Controller),可以根据 xml 配置或者注解进行查找。

第三步:处理器映射器 HandlerMapping 向前端控制器返回 Handler(Controller)

第四步:前端控制器调用处理器适配器(HandlerAdapter)去执行 Handler(Controller)

第五步:处理器适配器执行 Handler

第六步:Handler 执行完成后给适配器返回 ModelAndView

第七步:处理器适配器向前端控制器返回 ModelAndView

    ModelAndView 是SpringMVC 框架的一个底层对象,包括 Model(模型数据) 和 View(视图)

第八步:前端控制器请求试图解析器去进行视图解析

    根据逻辑视图名来解析真正的视图。

第九步:试图解析器向前端控制器返回 view

第十步:前端控制器进行视图渲染

    就是将模型数据(在 ModelAndView 对象中)填充到 request 域

第十一步:前端控制器向用户响应结果

用我自己的理解,结合servlet来说的话,就是:

首先页面发送一个请求,被SpringMVC的DispatcherServlet(前端控制器,即中央处理器)截获,就相当于传统servlet中配置的web.xml中配置的各种servlet,然后DispatcherServlet会根据URL发送给处理器映射器(Handler Mapping),就相当于servlet中的servlet-mapping中的作用,然后找到对应的Handler,即找到servlet中对应的类的地址,然后再给DispatcherServlet,DispatcherServlet根据Handler去找到对应的处理器适配器,就是找到SpringMVC的控制层,也就是Controller,相当于servlet中的控制层,一般就是命名为servlet,也就是相当于Strtus2中的Action动作,然后处理器适配器经过service层,dao层获得数据等,返回一个ModelAndView给DispatcherServlet,DispatcherServlet再去请求视图解析器去进行视图的解析,解析完返回view给DispatcherServlet,然后渲染页面,然后DispatcherServlet响应给页面。

下面我们对上面出现的一些组件进行解释:

1、前端控制器DispatcherServlet不需要程序员开发,需要WEB.xml配置)。

作用:接收请求,响应结果,相当于转发器,中央处理器。有了DispatcherServlet减少了其它组件之间的耦合度。

2、处理器映射器HandlerMapping(不需要程序员开发)。

作用:根据请求的url查找Handler。

3、处理器适配器HandlerAdapter(不需要程序员开发)。适配器模式

作用:按照特定规则(HandlerAdapter要求的规则)去执行Handler(Controller)。

4、处理器Handler(Controller)(需要程序员开发)。

注意:编写Handler时按照HandlerAdapter的要求去做,这样适配器才可以去正确执行Handler

5、视图解析器ViewResolver(不需要程序员开发,只需要配置)。

作用:进行视图解析,根据逻辑视图名解析成真正的视图(view)

6、视图View(需要程序员开发jsp)。标签显示页面数据

  注意:View是一个接口,实现类支持不同的View类型(jsp、freemarker、pdf…)

ps:不需要程序员开发的,需要程序员自己做一下配置即可。

可以总结出:需要我们开发的工作只有处理器 Handler 的编写以及视图比如JSP页面的编写。可能你还对诸如前端控制器、处理器映射器等等名词不太理解,那么接下来我们对其进行详细的介绍。

三、配置

①配置前端控制器(DispatcherServlet)

在 web.xml 文件中进行如下配置:

<!-- 配置前端控制器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来解析
      -->
    <url-pattern>/</url-pattern>
  </servlet-mapping>

②配置处理器适配器(不需要程序员配置)

在 springmvc.xml 文件中配置。用来约束我们所需要编码的 Handler类。

第一种配置:编写 Handler 时必须要实现 Controller

我们可以查看源码:

springmvc项目如何配置dbcp_spring_02

springmvc项目如何配置dbcp_spring_03

第二种配置:编写 Handler 时必须要实现 HttpRequestHandler

springmvc项目如何配置dbcp_MVC_04

springmvc项目如何配置dbcp_spring_05

③编写Handler(Controller)

在 springmvc.xml 文件中配置。通俗来讲,就是请求的 URL 到我们这里所编写的 Handler 类的某个方法进行一些业务逻辑处理。

  我们在上面讲解了两个处理器适配器来约束 Handler,那么我们就通过上面两种配置分别编写两个 Handler

  1.第一种:实现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;
	}
}

  2.第二种:实现 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字符串");

实际开发,我们加注解即可

④配置处理器映射HandlerMapping

在 springmvc.xml 文件中配置。通俗来讲就是请求的 URL 怎么能被 SpringMVC 识别,从而去执行我们上一步所编写好的 Handler

第一种:

<!-- 配置Handler -->
<bean name="/hello" class="com.bruceliu.controller.HelloController" />

<!-- 配置处理器映射器 将bean的name作为url进行查找,需要在配置Handler时指定bean name(就是url) -->
<bean class="org.springframework.web.servlet.handler.BeanNameUrlHandlerMapping" />

这样配置的话,那么请求的 URL,必须为 http://localhost:8080/项目名/hello

第二种方法:

<!-- 配置Handler -->
	<bean id="hello1" class="com.bruceliu.controller.HelloController" />
	<bean id="hello2" class="com.bruceliu.controller.HelloController2" />
	
	<!-- 第二种方法:简单URL配置处理器映射器 -->
	<bean class="org.springframework.web.servlet.handler.SimpleUrlHandlerMapping">
		<property name="mappings">
			<props>
				<prop key="/hello1">hello1</prop>
				<prop key="/hello2">hello2</prop>
			</props>
		</property>
	</bean>

这种配置请求的 URL可以为 http://localhost:8080/项目名/hello1,或者http://localhost:8080/项目名/hello2

总结:上面两种处理器映射器配置可以并存,前端控制器会正确的去判断 url 用哪个 Handler 去处理。

⑤配置视图解析器

1.第一种配置:

springmvc项目如何配置dbcp_MVC_06

如果这样配,那么在 Handler 中返回的必须是  路径+jsp页面名称+".jsp"

2.第二种配置

<!--配置视图解析器 -->
	<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 页面名就可以了

⑥DispatcherServlet.properties

上面我们讲解了各种配置,可能有人会问这么多配置,万一少配置了一样,那不就不能运行了,那我们能不能不配置呢?答案是肯定的,SpringMVC 给我们提供了一个 DispatcherServlet.properties 文件。系统会首先加载这里面的配置,如果我们没有配置,那么就默认使用这个文件的配置;如果我们配置了,那么就优先使用我们手动配置的。

springmvc项目如何配置dbcp_xml_07

在 SpringMVC 运行之前,会首先加载 DispatcherServlet.properties 文件里面的内容,那么我们来看看这里面都是什么。 在 SpringMVC 运行之前,会首先加载 DispatcherServlet.properties 文件里面的内容,那么我们来看看这里面都是什么

springmvc项目如何配置dbcp_xml_08

springmvc项目如何配置dbcp_springmvc项目如何配置dbcp_09

我们可以从上面得出,如果我们不手动进行各种配置,那么也有会默认的

  1、处理器适配器默认:org.springframework.web.servlet.mvc.HttpRequestHandlerAdapter

  2、处理器映射器默认:org.springframework.web.servlet.handler.BeanNameUrlHandlerMapping

       3、视图解析器默认:org.springframework.web.servlet.view.InternalResourceViewResolver

⑦Spring配置 <context:component-scan/> <mvc:annotation-driven />

1.<annotaion-driven/> 标签

这个标签对应的实现类是org.springframework.web.servlet.config.AnnotationDrivenBeanDefinitionParser

 

仔细阅读它的注释文档可以很明显的看到这个类的作用。解析这个文档:

这个类主要注册8个类的实例:

• 1.RequestMappingHandlerMapping   @RequestMapping
• 2.BeanNameUrlHandlerMapping
• 3.RequestMappingHandlerAdapter
• 4.HttpRequestHandlerAdapter   @controller
• 5.SimpleControllerHandlerAdapter
• 6.ExceptionHandlerExceptionResolver
• 7.ResponseStatusExceptionResolver
• 8.DefaultHandlerExceptionResolver

1是处理@RequestMapping注解的,2.将controller类的名字映射为请求url。1和2都实现了HandlerMapping接口,用来处理请求映射。

3是处理@Controller注解的控制器类,4是处理继承HttpRequestHandlerAdapter类的控制器类,5.处理继承SimpleControllerHandlerAdapter类的控制器。所以这三个是用来处理请求的。具体点说就是确定调用哪个controller的哪个方法来处理当前请求。

6,7,8全部继承AbstractHandlerExceptionResolver,这个类实现HandlerExceptionResolver,该接口定义:接口实现的对象可以解决处理器映射、执行期间抛出的异常,还有错误的视图。

所以<annotaion-driven/>标签主要是用来帮助我们处理请求映射,决定是哪个controller的哪个方法来处理当前请求,异常处理。

2.<context:component-scan/>标签

它的实现类是org.springframework.context.annotation.ComponentScanBeanDefinitionParser.

把鼠标放在context:component-scan上就可以知道有什么作用的,用来扫描该包内被@Repository @Service @Controller的注解类,然后注册到工厂中。并且context:component-scan激活@ required。@ resource,@ autowired、@PostConstruct @PreDestroy @PersistenceContext @PersistenceUnit。使得在适用该bean的时候用@Autowired就行了。

四、Spring和SpringMVC的关系

  • pingMvc是Spring体系中的一个Web模块。负责Web控制层工作!
  • SpringMvc依赖于Spring存在!
  • Spring和SpringMvc是一个父子容器关系!

springmvc项目如何配置dbcp_MVC_10