SpringMVC基础

2016-12-31  本文已影响268人  泊浮目

Spring MVC工作原理

SpringMVCExecuteStep.jpg
  1. 在请求离开浏览器时,会带有用户所请求内容的信息,至少会包含请求的URL。但是还可能带有其他的信息,例如用户提交的表单信息。

  2. 请求旅程的第一站是Spring的DispatcherServlet。与大多数基于Java的Web框架一样,Spring MVC所有的请求都会通过一个前端控制器(front controller)Servlet。前端控制器是常用的Web应用程序模式,在这里一个单实例的Servlet将请求委托给应用程序的其他组件来执行实际的处理。在Spring MVC中,DispatcherServlet就是前端控制器。

  3. DispatcherServlet的任务是将请求发送给Spring MVC控制器(controller)。控制器是一个用于处理请求的Spring组件。在典型的应用程序中可能会有多个控制器,DispatcherServlet需要知道应该将请求发送给哪个控制器。所以DispatcherServlet以会查询一个或多个处理器映射(handler mapping)2来确定请求的下一站在哪里。处理器映射会根据请求所携带的URL信息来进行决策。

  4. 一旦选择了合适的控制器,DispatcherServlet会将请求发送给选中的控制器。到了控制器,请求会卸下其负载(用户提交的信息)并耐心等待控制器处理这些信息。(实际上,设计良好的控制器本身只处理很少甚至不处理工作,而是将业务逻辑委托给一个或多个服务对象进行处理。)

  5. 控制器在完成逻辑处理后,通常会产生一些信息,这些信息需要返回给用户并在浏览器上显示。这些信息被称为模型(model)。不过仅仅给用户返回原始的信息是不够的——这些信息需要以用户友好的方式进行格式化,一般会是HTML。所以,信息需要发送给一个视图(view),通常会是JSP。

  6. 控制器所做的最后一件事就是将模型数据打包,并且标示出用于渲染输出的视图名。它接下来会将请求连同模型和视图名发送回DispatcherServlet。

  7. 这样,控制器就不会与特定的视图相耦合,传递给DispatcherServlet的视图名并不直接表示某个特定的JSP。实际上,它甚至并不能确定视图就是JSP。相反,它仅仅传递了一个逻辑名称,这个名字将会用来查找产生结果的真正视图。DispatcherServlet将会使用视图解析器(view resolver)来将逻辑视图名匹配为一个特定的视图实现,它可能是也可能不是JSP。

  8. 既然DispatcherServlet已经知道由哪个视图渲染结果,那请求的任务基本上也就完成了。它的最后一站是视图的实现(可能是JSP),在这里它交付模型数据。请求的任务就完成了。视图将使用模型数据渲染输出,这个输出会通过响应对象传递给客户端(不会像听上去那样硬编码)。


实际上SpringMVC的使用非常简单,我们只要扩展一个路径映射关系;定义一个视图解析器;再定义一个业务逻辑的处理流程规则,SpringMVC就能狗帮你完成所有的MVC的功能了。要搞清楚SpringMVC如何工作,主要还是看DispatcherServlet的代码。与DispatcherServlet类相关的结构图。

DispatcherServletClass.jpg

其中DispatcherServlet继承了HttpServlet,在Servlet的init方法调用时DispatcherServlet执行SpringMVC的初始化工作。DispatcherServlet初始化什么,可以在其initStrategies方法中知道:

protected void initStrategies(ApplicationContext context){
  initMultipartResovler(context);
  initLocaleResolver(context);
  initThemeResolver(context);
  initHandlerMappings(context);
  initHandlerAdapters(context);
  initHandlerExceptionResolvers(context);
  initRequestToViewNameTranslator(context);
  initViewResovlers(context);
}
  1. initMultipartResovler:初始化MultipartResover,用于处理文件上传服务,当有文件上传时,将HttpServletRequest封装成DefaultMultipartHttp,将每个上传的内容封装成CommonsMultipartFile。
  2. initLocaleResolver:用于处理国际化问题,通过解析请求的Locle和设置响应的Locale来控制应用中的字符编码问题。
  3. initThemeResolver:用于定义一个主题,可根据用户喜好来设置用户访问的页面的样式,可以将这个样式作为一个Theme Name保存,保存在用于请求的Cookie中或者保存在服务端的Session中,以后每次请求根据这个Theme Name返回特定的内容。
  4. initHandlerMappings:定义请求和处理Handler之间的关系,如果没有定义,则获取默认的两个,BeanNameURLHandlerMapping和DefaultAnnotationHandlerMapping
  5. initHandlerAdapters:根据Handler定义类型不同定义不同的处理规则
  6. initHandlerExceptionResolvers:当Handler处理出错的时候,通过这个Handler来统一处理,默认的实现类是SimpleMappingExceptionResolver,将错误日志记录在log文件中,并且转到默认的错误页面。
  7. initRequestToViewNameTranslator:将指定的ViewName按照定义的RequestToViewNameTranslator替换成想要的格式。
  8. initViewResovlers:将view解析成页面,在ViewResolver中可以设置多个解析策略,如可以根据JSP解析,或者按照Velocity模板解析。默认的解析策略是InternalResourceViewResolver,按照JSP页面来解析。

在SpringMVC框架中,有3个组件是用户必须要定义和扩展的:定义URL映射规则、实现业务逻辑的Handler实例对象、模板渲染资源。而连接Handler实例对象和模板渲染的纽带就是Model模型了。

再来看看DispatcherServlet启动时都做了哪些事情?

Spring MVC常用注解

@Contorller

@Contorller注解在类上,表明这个类是Spring MVC里的Controller,将其声明为Spring的一个Bean,Dispatcher Servlet会自动扫描注解了此注解的类,并将Web请求映射到注解了@RequestMapping的方法上。

@RequestMapping

@RequestMapping 注解是用来映射Web请求(访问路径和参数)、处理类和方法的。@RequestMapping 可注解在类或方法上。注解在方法上的@RequestMapping路径会继承注解在类的路径,@RequestMapping支持Servlet的request和response作为参数,也支持对request和response的媒体类型进行配置。

@ResponseBody

@ResponseBody 支持讲返回值放在response体内,而不是返回一个页面。我们在很多基于Ajax的程序的时候,可以以此注解返回数据而不是页面;此注解可放置在返回值或者方法上。

@RequestBody

@RequestBody 允许request的参数在request体内,而不是直接链接在地址后面。此注解放置在参数前。

@PathVaribale

@PathVaribale 用来接收路径参数,如/news/0001,可接收001作为参数,此注解放置在参数前。

@RestContorller

@RestContorller 是一个组合注解,组合了@Controller和@ResponseBody,这就意味着当你只开发一个和页面交互数据的控制的时候,需要使用此注解。若没有此注解,要想实现上述功能,则需自己在代码中@Controller和@ResponseBody两个注解。

静态资源映射

拦截器

@ControllerAdvice

通过@ControllerAdvice,我们可以将对于控制器的全局配置放置在同一个位置,注解了@Contorller的类的法官法可使用@ExceptionHandler、@InitBinder、@ModelAttribute注解到方法上,这对所有注解了@RequestMapping的控制器内的方法有效。

@快捷的ViewController

配置页面转向可以使用来自WebMvcConfigurerAdpter的addViewControllers方法。示例:

addViewControllers(ViewControllerRegistry registry){
  registry.addViewControllers("/PATH").setViewName("/VIEW-NAME")
}
上一篇下一篇

猜你喜欢

热点阅读