一、注解驱动控制器
在上面的示例中,是通过BeanNameUrlHandlerMapping的方式完成请求与controller之间的映射关系。
如果有多个请求时,需要配置多个映射关系,并建立多个Controller来进行请求处理,实现繁琐,如何解决?
解决方式是使用SpringMVC提供的基于注解的处理器:
也就是上面的
<mvc:annotation-driven/>
标签,这个标签也就是一个基于注解的一个处理器,使用注解的方式来进行springmvc的开发。
使用方式:
首先我们需要去更改springmvc的处理器映射,配置从原来的beannameurlmapping更改为现在的支持注解的这个标签。配置这个标签之后,springmvc会帮助我们自动做一些组件注册之类的事情,比如上图中该标签下面的两个组件。注册之后,springmvc会通过这两个bean的实例来完成对注解的支持,比如下面的@Controller和@RequestMapping。从而可以找出请求的url和handller也就是处理器的响应的方法之间的关系,进行一些相应的关联。
对于Controller注解,如果一个javaBean没有标注这个注解,呢他仅仅是一个普通的javabean,如果标注了,则它就可以作为一个处理请求的Controller控制器。
对于requestMapping注解,他是对一个请求的一个映射,可以加在类名的上面,也可以加在控制器的处理方法的上面,从而建立与url的请求的映射关系。
这些注解想要进行生效,则必须加入这两个注解上面的那个标签。
二、示例代码
配置文件:
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:p="http://www.springframework.org/schema/p"
xmlns:context="http://www.springframework.org/schema/context"
xmlns:mvc="http://www.springframework.org/schema/mvc"
xsi:schemaLocation="
http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context.xsd
http://www.springframework.org/schema/mvc
http://www.springframework.org/schema/mvc/spring-mvc.xsd">
<!-- 配置处理器映射 HandlerMapping:将一个url请求指定给相应的controller去处理 -->
<!-- <bean name="/index.html" class="cn.smbms.controller.IndexController"></bean> -->
<mvc:annotation-driven></mvc:annotation-driven>
<context:component-scan base-package="cn.smbms.controller"></context:component-scan>
<!-- 配置视图解析器 /WEB-INF/jsp/index.jsp-->
<bean class="org.springframework.web.servlet.view.InternalResourceViewResolver">
<property name="prefix" value="/WEB-INF/jsp/"></property>
<property name="suffix" value=".jsp"></property>
</bean>
</beans>
控制器:
package cn.smbms.controller;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.servlet.ModelAndView;
import org.springframework.web.servlet.mvc.AbstractController;
@Controller
public class IndexController {
@RequestMapping("/index")
public ModelAndView index() {
System.out.println("hello, springmvc");
return new ModelAndView("index");
}
}
运行结果正常。
同时因为此时返回的是逻辑视图名,所以控制器也可以写成下面这样:
package cn.smbms.controller;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.servlet.ModelAndView;
import org.springframework.web.servlet.mvc.AbstractController;
@Controller
public class IndexController {
@RequestMapping("/index")
public String index() {
System.out.println("hello, springmvc");
return "index";
}
}
运行结果正常。
三、springmvc框架的请求处理流程
springmvc是一个基于请求驱动的web框架,他也使用了一些前端控制器的模式来进行设计,然后根据请求映射的规则来分发给相应的处理器来进行业务功能的处理。
其具体处理流程如下:
首先,用户发送请求给前端控制器,前端控制器根据这个请求比如说url来决定要选择哪个处理器来进行相应的处理,这个处理器接受到请求之后则会进行业务的处理,这个时候则会调用相应的业务对象来进行业务处理。业务对象处理完了之后会返回一个处理结果,也就是模型数据。处理器拿到这个数据模型之后,再把他返回给前端控制器,这时候是一个ModelAndView对象。前端控制器得到这个结果之后,则会把控制权收回,然后根据返回的这个逻辑视图名选择相应的真正的视图,然后把模型数据传入到视图里面,进行视图渲染的展示。最后这个前端控制器再次收回控制权,将响应的结果返回给客户端。这就是整个流程。在这个流程中,处于核心的是前端控制器,它不断地组织和协调各部门的工作,最终再把结果返回给结果。这个前端控制器其实就是我们最开始配置的dispatcherServlet。
四、SpringMVC体系结构
在上图中,用户发送请求到DispatcherServlet(也不是所有的请求,到底有哪些请求可以在web.xml里面配置),接收到这些请求之后,他会把这些工作委派给其他的组件来进行处理,其他组件完成之后再把结果返回给他,他最后再把结果进行渲染,再输出返回给用户。
更细致的流程如下:
首先,客户端发出http请求,web服务器接受请求,如果匹配dispatcherServlet的话,web服务器会把这个请求交给dispatcherServlet去处理。dispatcherServlet接受到请求之后,会根据请求的信息(不仅仅是url)通过HandlerMapping的配置去找到处理请求的handler,这个处理器映射的配置是在servletName-servlet.xml中完成的。通过HandlerMapping,dispatcherServlet找到相应的处理器Handler,然后再通过HandlerAdapter对这个Handler进行封装,然后再通过统一的适配器接口调用相应的Handler(这个HandlerAdapter可以理解为具体使用Handler干活的人)。在请求信息到达处理的Handler之前,在这段时间里,springmvc框架还完成了很多工作,他会将请求信息以一定的方式进行转换(就是上图中右下角红框框着的那部分),并绑定到这种请求的入参中,入参的对象会进行数据转换,数据的格式化以及数据验证等等。等到框架做完这些之后,请求才会到达最终的Handler里面,进行相应的业务逻辑的处理。在这个地方一直提到的handler和处理器、控制器等等这些可以理解为你项目中所写的这个controller。这些处理完成之后返回一个ModelAndView对象,这个对象包含逻辑视图名和数据模型给DispatcherServlet。DispatcherServlet此时会把控制权收回,此时ModelAndView里面包含的逻辑视图名并非真正的视图对象,这个时候DispatcherServlet会通过ViewResolver这个组件将逻辑视图名解析为一个真正的视图对象。当然负责进行视图展示的格式有多种多样,包括jsp、xml、pdf等等。得到真正的视图对象之后,ModelAndView里面还有一个Model,还可以使用ModelAndView里面的模型数据对View再次进行视图渲染,最终把结果返回给客户端,最终显示出来的可能是一个html页面,也可能是xml,json等等。
4.1DispatcherServlet(前端控制器)
是SpringMVC中最核心的一个类,主要负责协调组织不同的组件来完成请求处理并且返回响应。在配置的时候最早配置的就是这个,在web.xml中进行配置。
4.2Handler(处理器)
对应MVC中的Controller层。
ModelAndView对象包含了两部分内容,一部分是视图界面呈现时所需要使用的模型对象,另一部分是逻辑视图名。由于java一次只能返回一个对象,所以ModelAndView的作用就是把这两个对象进行了封装,以方便一次性返回我们所需要的Model和View,当然返回的是模型还是视图也是可以选的,在一些情况下,模型里面没有任何数据,此时就返回视图就可以了,比如说之前的hellospring示例。
处理器映射
简单的理解就是根据请求的url来进行相应的映射,其接口的实现有很多,默认的是BeanNameUrlHandlerMapping,我们在第一个示例中就是使用的这个,当你什么都不配置的时候,SpringMVC就会默认使用这个,将请求的URl映射到同名的控制器Bean上。但是这种做法存在弊端,就是我需要进行多个处理器映射的配置。
应用最广泛的就是DefaultAnnotationHandlerMapping,这种只需要将请求映射到标注@Requestmapping注解的控制器和处理方法上。之前示例在使用的时候,配置文件中增加了一键式配置,他会自动的帮我们注册DefaultAnnotationHandlerMapping。
还有一个是RequestMappingHandlerMapping,这个跟DefaultAnnotationHandlerMapping一样,在3.2版本之后,也可以使用这个,这个知道就可以了。
适配器
HandlerAdapter,这个主要是对Handler进行一个封装,可以把他简单地理解为他就是使用Handler干活的人。里面有两个实现:AnnotationMethodHandlerAdapter和RequestMappingHandlerAdapter。AnnotationMethodHandlerAdapter和上面的DefaultAnnotationHandlerMapping这两个是一起的;RequestMappingHandlerAdapter是和RequestMappingHandlerMapping一起的。
视图解析器
ViewResolver,主要使用InternalResourceView,作为jsp视图的一个解析。使用这个将一个逻辑视图名解析为一个真正的视图对象。