一、spring mvc处理请求的流程。
客户端发出http请求,如果匹配到DispatcherServlet的请求映射路径,则web容器将请求转交给DispatcherServlet处理。
DispatcherServlet接受到请求后,根据请求的信息和HandlerMapping的配置,找到处理请求的处理器(Handler)。HandlerMapping类是路由控制器。
得到相对应的Handler后,通过HandlerAdapter对Handler进行封装,再以统一的适配器接口调用Handler。
处理完业务后,返回ModelAndView给DispatcherServlet。
通过ViewResolver完成ModelAndView逻辑视图到真实视图的解析工作。
二、在web中配置DispatcherServlet
1.配置
contextConfigLocation classpath:spring-base.xml,classpath:spring-another.xml
通过contextConfigLoaction在web.xml里配置需要加载的spring配置文件
2.配置
org.springframework.web.context.ContextLoaderListener
在web.xml里配置Listener,本Listener会去加载第一步中指定的配置文件,如果指定了要加载的文件,则会去加载相应的xml,而不会去加载/WEB-INF/下的applicationContext.xml。(指定加载即是第1步的配置)如果没有指定的话,默认会去/WEB-INF/下加载applicationContext.xml。
3.配置DispatcherServlet
Servlet org.springframework.web.servlet.DispatcherServlet 1 Servlet *.do
配置了名为Servlet的DispatcherServlet,它默认自动加载/WEB-INF/Servlet-servlet.xml的spring配置文件,启动web层的Spring容器。
可以通过
属性namespace:对于DispatcherServlet的命名空间,默认为
属性contextConfigLocation:如果DispatcherServlet对应上下文有多个spring配置文件,则可以使用该属性指定。
三、注解驱动的控制器
1.使用@RequestMapping映射请求
/** * 接收GET请求 * @return */ @RequestMapping(value="/delete",method = RequestMethod.GET) public String get(){ System.out.println("get"); return "get"; } /** * 接收POST请求 * @return */ @RequestMapping(value="/delete",method = RequestMethod.POST) public String post(){ System.out.println("post"); return "post"; } /** * 只接收 name 参数 * @return */ @RequestMapping(value="/delete",params="name") public String params(String name){ 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") public String headers(){ System.out.println("headers"); return "helloworld"; }
2.请求处理方法签名
Spring MVC通过分析处理方法的签名,把http请求信息绑定到处理方法的相应入参中,然后再调用处理方法得到返回值。以下是四种绑定参数的方式:
// ①请求参数按名称匹配的方式绑定到方法入参中,方法返回对应的字符串代表逻辑视图名 @RequestMapping(value = "/handle1") public String handle1(@RequestParam("userName") String userName, @RequestParam("password") String password, @RequestParam("realName") String realName) { return "success"; } // ②将Cooke值及报文头属性绑定到入参中、方法返回ModelAndView @RequestMapping(value = "/handle2") public ModelAndView handle2(@CookieValue("JSESSIONID") String sessionId, @RequestHeader("Accept-Language") String accpetLanguage) { ModelAndView mav = new ModelAndView(); mav.setViewName("success"); mav.addObject("user", new User()); return mav; } // ③请求参数按名称匹配的方式绑定到user的属性中、方法返回对应的字符串代表逻辑视图名 @RequestMapping(value = "/handle3") public String handle3(User user) { return "success"; } // ④直接将HTTP请求对象传递给处理方法、方法返回对应的字符串代表逻辑视图名 @RequestMapping(value = "/handle4") public String handle4(HttpServletRequest request) { return "success"; }
3.使用HttpMessageConverter
1)概念:HttpMessageConverter
2)使用:通过@RequestBody和@ResponseBody
@RequestMapping(value = "/handle41") public String handle41(@RequestBody String body) {//把请求报文体转换为字符串绑定到RequestBody中 System.out.println(body); return "success"; } @ResponseBody//把数据输出到响应流中。 @RequestMapping(value = "/handle42/{p_w_picpathId}") public byte[] handle42(@PathVariable("p_w_picpathId") String p_w_picpathId) throws IOException { System.out.println("load p_w_picpath of " + p_w_picpathId); Resource res = new ClassPathResource("/p_w_picpath.jpg"); byte[] fileData = FileCopyUtils.copyToByteArray(res.getInputStream()); return fileData; }
@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") public String handle63(ModelMap modelMap) { User user = (User) modelMap.get("user");//springmvcv会把隐含模型对象传递给modelMap,所以可以访问模型的数据 user.setUserName("tom"); modelMap.addAttribute("testAttr", "value1"); return "/user/showUser"; }
四、处理方法的数据绑定
spring mvc通过反射机制对目标处理函数的签名进行分析,将请求消息绑定到处理方法的入参中,核心部件是DataBinder。
1.数据格式化
spring使用转换器进行源类型对象到目标类型对象的转换。spring提供两个内建的实现类,分别支持数字及日期类型的注解驱动格式化。
@NumberFormat和@DateTimeFormat注解
@DateTimeFormat(pattern="yyyy-MM-dd") private Date birthday; @NumberFormat(pattern="#,###.##") private long salary;
form.add("birthday", "1980-01-01"); form.add("salary", "4,500.00");
2.数据校验
1)在model设置检验条件
@Pattern(regexp="w{4,30}")//通过正则校验,匹配4-30个包含数字、字母、下划线的字符 private String userName; @Pattern(regexp="S{6,30}")//通过正则校验,匹配6-30个非空白的字符 private String password; @Length(min=2,max=100)//长度限制在2-100 private String realName; @Past //时间值必须是个过去的时间 @DateTimeFormat(pattern="yyyy-MM-dd") private Date birthday; @DecimalMin(value="1000.00")//数据必须在1000.00-100000.00之间 @DecimalMax(value="100000.00") @NumberFormat(pattern="#,###.##") private long salary;
2)然后在处理函数中处理
@RequestMapping(value = "/handle91") public String handle91(@Valid @ModelAttribute("user") User user, BindingResult bindingResult, ModelMap mm) { 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中配置试图解析器