一、spring mvc处理请求的流程。

客户端发出http请求,如果匹配到DispatcherServlet的请求映射路径,则web容器将请求转交给DispatcherServlet处理。

DispatcherServlet接受到请求后,根据请求的信息和HandlerMapping的配置,找到处理请求的处理器(Handler)。HandlerMapping类是路由控制器。

得到相对应的Handler后,通过HandlerAdapter对Handler进行封装,再以统一的适配器接口调用Handler。

处理完业务后,返回ModelAndView给DispatcherServlet。

通过ViewResolver完成ModelAndView逻辑视图到真实视图的解析工作。



二、在web中配置DispatcherServlet

1.配置<context-param>

<context-param><param-name>contextConfigLocation</param-name><param-value>classpath:spring-base.xml,classpath:spring-another.xml</param-value></context-param>

通过contextConfigLoaction在web.xml里配置需要加载的spring配置文件


2.配置<listener>

<listener><listener-class>org.springframework.web.context.ContextLoaderListener</listener-class></listener>

在web.xml里配置Listener,本Listener会去加载第一步中指定的配置文件,如果指定了要加载的文件,则会去加载相应的xml,而不会去加载/WEB-INF/下的applicationContext.xml。(指定加载即是第1步的配置)如果没有指定的话,默认会去/WEB-INF/下加载applicationContext.xml。


3.配置DispatcherServlet

<servlet><servlet-name>Servlet</servlet-name><servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class><load-on-startup>1</load-on-startup></servlet><servlet-mapping><servlet-name>Servlet</servlet-name><url-pattern>*.do</url-pattern></servlet-mapping>

配置了名为Servlet的DispatcherServlet,它默认自动加载/WEB-INF/Servlet-servlet.xml的spring配置文件,启动web层的Spring容器。

可以通过<servlet>的<init-param>指定:

属性namespace:对于DispatcherServlet的命名空间,默认为<servlet-name>-servlet。用于构造spring配置文件的路径,指定该属性,可以指定spring配置文件。

属性contextConfigLocation:如果DispatcherServlet对应上下文有多个spring配置文件,则可以使用该属性指定。


三、注解驱动的控制器

1.使用@RequestMapping映射请求

/***接收GET请求*@return*/@RequestMapping(value="/delete",method=RequestMethod.GET)publicStringget(){System.out.println("get");return"get";}/***接收POST请求*@return*/@RequestMapping(value="/delete",method=RequestMethod.POST)publicStringpost(){System.out.println("post");return"post";}/***只接收name参数*@return*/@RequestMapping(value="/delete",params="name")publicStringparams(Stringname){System.out.println("hello"+name);return"helloworld";}/***只接收请求头中Content-Type为text/html;charset=UTF-8的请求*@return*/@RequestMapping(value="/delete",headers="Content-Type:text/html;charset=UTF-8")publicStringheaders(){System.out.println("headers");return"helloworld";}

2.请求处理方法签名

Spring MVC通过分析处理方法的签名,把http请求信息绑定到处理方法的相应入参中,然后再调用处理方法得到返回值。以下是四种绑定参数的方式:

//①请求参数按名称匹配的方式绑定到方法入参中,方法返回对应的字符串代表逻辑视图名@RequestMapping(value="/handle1")publicStringhandle1(@RequestParam("userName")StringuserName,@RequestParam("password")Stringpassword,@RequestParam("realName")StringrealName){return"success";}//②将Cooke值及报文头属性绑定到入参中、方法返回ModelAndView@RequestMapping(value="/handle2")publicModelAndViewhandle2(@CookieValue("JSESSIONID")StringsessionId,@RequestHeader("Accept-Language")StringaccpetLanguage){ModelAndViewmav=newModelAndView();mav.setViewName("success");mav.addObject("user",newUser());returnmav;}//③请求参数按名称匹配的方式绑定到user的属性中、方法返回对应的字符串代表逻辑视图名@RequestMapping(value="/handle3")publicStringhandle3(Useruser){return"success";}//④直接将HTTP请求对象传递给处理方法、方法返回对应的字符串代表逻辑视图名@RequestMapping(value="/handle4")publicStringhandle4(HttpServletRequestrequest){return"success";}


3.使用HttpMessageConverter

1)概念:HttpMessageConverter<T>负责把请求信息转换为一个对象,将对象输出作为响应信息。

2)使用:通过@RequestBody和@ResponseBody

@RequestMapping(value="/handle41")publicStringhandle41(@RequestBodyStringbody){//把请求报文体转换为字符串绑定到RequestBody中System.out.println(body);return"success";}@ResponseBody//把数据输出到响应流中。@RequestMapping(value="/handle42/{p_w_picpathId}")publicbyte[]handle42(@PathVariable("p_w_picpathId")Stringp_w_picpathId)throwsIOException{System.out.println("loadp_w_picpathof"+p_w_picpathId);Resourceres=newClassPathResource("/p_w_picpath.jpg");byte[]fileData=FileCopyUtils.copyToByteArray(res.getInputStream());returnfileData;}

@RequestBody

作用:

i) 该注解用于读取Request请求的body部分数据,使用系统默认配置的HttpMessageConverter进行解析,然后把相应的数据绑定到要返回的对象上;

ii) 再把HttpMessageConverter返回的对象数据绑定到 controller中方法的参数上。

@ResponseBody

作用:

该注解用于将Controller的方法返回的对象,通过适当的HttpMessageConverter转换为指定格式后,写入到Response对象的body数据区。

使用时机:

返回的数据不是html标签的页面,而是其他某种格式的数据时(如json、xml等)使用;

@RestController

作用:

相当于@Controller+@ResponseBody


4.处理模型数据

spring mvc在调用处理方法前会创建一个隐含的模型对象,如果处理方法的入参为Map或者Model类型,则spring mvc会把隐形模型的引用传递给这些入参,一般是在方法参数写上Model model,然后通过model.addAttribute(key,value)来添加数据。

@RequestMapping(value="/handle63")publicStringhandle63(ModelMapmodelMap){Useruser=(User)modelMap.get("user");//springmvcv会把隐含模型对象传递给modelMap,所以可以访问模型的数据user.setUserName("tom");modelMap.addAttribute("testAttr","value1");return"/user/showUser";}


四、处理方法的数据绑定
spring mvc通过反射机制对目标处理函数的签名进行分析,将请求消息绑定到处理方法的入参中,核心部件是DataBinder。

1.数据格式化

spring使用转换器进行源类型对象到目标类型对象的转换。spring提供两个内建的实现类,分别支持数字及日期类型的注解驱动格式化。

@NumberFormat和@DateTimeFormat注解

<beanid="conversionService"class="org.springframework.format.support.FormattingConversionServiceFactoryBean"><propertyname="converters"><list><beanclass="com.smart.domain.StringToUserConverter"/></list></property></bean>

@DateTimeFormat(pattern="yyyy-MM-dd")privateDatebirthday;@NumberFormat(pattern="#,###.##")privatelongsalary;

form.add("birthday","1980-01-01");form.add("salary","4,500.00");

2.数据校验

<mvc:annotation-driven/>会默认装配一个LocalValidatorFactoryBean,通过处理方法的入参上标注的@Valid注解,即可让spring mvc在完成数据绑定后执行数据校验工作。

1)在model设置检验条件

@Pattern(regexp="w{4,30}")//通过正则校验,匹配4-30个包含数字、字母、下划线的字符privateStringuserName;@Pattern(regexp="S{6,30}")//通过正则校验,匹配6-30个非空白的字符privateStringpassword;@Length(min=2,max=100)//长度限制在2-100privateStringrealName;@Past//时间值必须是个过去的时间@DateTimeFormat(pattern="yyyy-MM-dd")privateDatebirthday;@DecimalMin(value="1000.00")//数据必须在1000.00-100000.00之间@DecimalMax(value="100000.00")@NumberFormat(pattern="#,###.##")privatelongsalary;

2)然后在处理函数中处理

@RequestMapping(value="/handle91")publicStringhandle91(@Valid@ModelAttribute("user")Useruser,BindingResultbindingResult,ModelMapmm){if(bindingResult.hasErrors()){return"/user/register3";}else{return"/user/showUser";}}

在入参对象前添加了@Valid注解,同时在其后声明了一个BindingResult入参。然后通过BindingResult就可以进行判断,处理了。BindingResult除了hasErrors()外还有以下几个接口:getFieldError(String field)获取属性名对应的校验错误、getFiledErrors()获取所有属性校验错误、getFiledValue(String field)获取属性值、getErrorCount()获取错误数量。


五、视图和视图解析器

请求处理方法执行完成后,最终返回一个ModelAndView对象,spring mvc借助视图解析器得到最终的视图对象View。视图对象是一个bean,由视图解析器负责实例化。

在xml中配置试图解析器

<beanclass="org.springframework.web.servlet.view.InternalResourceViewResolver"p:order="100"p:viewClass="org.springframework.web.servlet.view.JstlView"p:prefix="/WEB-INF/views/"p:suffix=".jsp"/>