首先了解MVC需要先知道SpringMVC是什么,三层架构和SpringMVC的mvc区别是什么
SpringMVC就是表现层,其实就是对servlet的封装,和servlet区别是:
-Model(模型):模型包含业务模型和数据模型,数据模型?于封装数据,业务模型?于处理业务
-View(视图): 通常指的就是我们的 jsp 或者 html。作??般就是展示数据的。通常视图是依据模型数据创建的
-Controller(控制器): 是应?程序中处理?户交互的部分。作??般就是处理程序逻辑的
开发过程
1)配置DispatcherServlet前端控制器
2)开发处理具体业务逻辑的Handler(@Controller、@RequestMapping)
3)xml配置?件配置controller扫描,配置springmvc三?件
4)将xml?件路径告诉springmvc(DispatcherServlet)
流程说明
第?步:?户发送请求?前端控制器DispatcherServlet
第?步:DispatcherServlet收到请求调?HandlerMapping处理器映射器
第三步:处理器映射器根据请求Url找到具体的Handler(后端控制器),?成处理器对象及处理器拦截器(如果 有则?成)?并返回DispatcherServlet
第四步:DispatcherServlet调?HandlerAdapter处理器适配器去调?Handler
第五步:处理器适配器执?Handler
第六步:Handler执?完成给处理器适配器返回ModelAndView
第七步:处理器适配器向前端控制器返回 ModelAndView,ModelAndView 是SpringMVC 框架的?个底层对 象,包括 Model 和 View
第?步:前端控制器请求视图解析器去进?视图解析,根据逻辑视图名来解析真正的视图。
第九步:视图解析器向前端控制器返回View
第?步:前端控制器进?视图渲染,就是将模型数据(在 ModelAndView 对象中)填充到 request 域
第??步:前端控制器向?户响应结果
HandlerMapping(处理器映射器)
HandlerMapping 是?来查找 Handler 的,也就是处理器,具体的表现形式可以是类,也可以是?法。?如,标注了@RequestMapping的每个?法都可以看成是?个Handler。Handler负责具体实际的请求处理,在请求到达后,HandlerMapping 的作?便是找到请求相应的处理器Handler 和 Interceptor.
HandlerAdapter(处理器适配器)
HandlerAdapter 是?个适配器。因为 Spring MVC 中 Handler 可以是任意形式的,只要能处理请求即可。但是把请求交给 Servlet 的时候,由于 Servlet 的?法结构都是doService(HttpServletRequest req,HttpServletResponse resp)形式的,要让固定的 Servlet 处理?法调? Handler 来进?处理,便是 HandlerAdapter 的职责。
HandlerExceptionResolver(异常解析器)
HandlerExceptionResolver ?于处理 Handler 产?的异常情况。它的作?是根据异常设置ModelAndView,之后交给渲染?法进?渲染,渲染?法会将 ModelAndView 渲染成??。
ViewResolver(视图解析器)
?于将String类型的视图名和Locale解析为View类型的视图,只有?个resolveViewName()?法。从?法的定义可以看出,Controller层返回的String类型视图名viewName 最终会在这?被解析成为View。View是?来渲染??的,也就是说,它会将程序返回的参数和数据填?模板中,?成html?件。ViewResolver 在这个过程主要完成两件事情:
ViewResolver 找到渲染所?的模板(第?件?事)和所?的技术(第?件?事,其实也就是找到视图的类型,如JSP)并填?参数。默认情况下,Spring MVC会?动为我们配置?个InternalResourceViewResolver,是针对 JSP 类型视图的。
RequestToViewNameTranslator
RequestToViewNameTranslator 组件的作?是从请求中获取 ViewName.因为 ViewResolver 根据ViewName 查找 View,但有的 Handler 处理完成之后,没有设置 View,也没有设置 ViewName,便要通过这个组件从请求中查找 ViewName。
LocaleResolver
ViewResolver 组件的 resolveViewName ?法需要两个参数,?个是视图名,?个是 Locale。LocaleResolver ?于从请求中解析出 Locale,?如中国 Locale 是 zh-CN,?来表示?个区域。这个组件也是 i18n 的基础。
ThemeResolver
ThemeResolver 组件是?来解析主题的。主题是样式、图?及它们所形成的显示效果的集合。Spring MVC 中?套主题对应?个 properties?件,??存放着与当前主题相关的所有资源,如图?、CSS样式等。创建主题?常简单,只需准备好资源,然后新建?个“主题名.properties”并将资源设置进去,放在classpath下,之后便可以在??中使?了。SpringMVC中与主题相关的类有ThemeResolver、ThemeSource和Theme。ThemeResolver负责从请求中解析出主题名,ThemeSource根据主题名找到具体的主题,其抽象也就是Theme,可以通过Theme来获取主题和具体的资源。
MultipartResolver
MultipartResolver ?于上传请求,通过将普通的请求包装成MultipartHttpServletRequest 来实现。MultipartHttpServletRequest 可以通过getFile() ?法 直接获得?件。如果上传多个?件,还可以调? getFileMap()?法得到Map<FileName,File>这样的结构,MultipartResolver 的作?就是封装普通的请求,使其拥有?件上传的功能。
FlashMapManager
FlashMap ?于重定向时的参数传递,?如在处理?户订单时候,为了避免重复提交,可以处理完post请求之后重定向到?个get请求,这个get请求可以?来显示订单详情之类的信息。这样做虽然可以规避?户重新提交订单的问题,但是在这个??上要显示订单的信息,这些数据从哪?来获得呢?因为重定向时么有传递参数这?功能的,如果不想把参数写进URL(不推荐),那么就可以通过FlashMap来传递。只需要在重定向之前将要传递的数据写?请求(可以通过ServletRequestAttributes.getRequest()?法获得)的属性OUTPUT_FLASH_MAP_ATTRIBUTE
中,这样在重定向之后的Handler中Spring就会?动将其设置到Model中,在显示订单信息的??上就可以直接从Model中获取数据。FlashMapManager 就是?来管理 FalshMap 的。
通常需要了解的是HandlerMapping,HandlerAdapter,HandlerExceptionResolver,MultipartResolver。其他的不常使用
1)程序先执?preHandle()?法,如果该?法的返回值为true,则程序会继续向下执?处理器中的?法,否则将不再向下执?。
2)在业务处理器(即控制器Controller类)处理完请求后,会执?postHandle()?法,然后会通过DispatcherServlet向客户端返回响应。
3)在DispatcherServlet处理完请求后,才会执?afterCompletion()?法。
多个拦截器流程
从图可以看出,当有多个拦截器同时?作时,它们的preHandle()?法会按照配置?件中拦截器的配置顺序执?,?它们的postHandle()?法和afterCompletion()?法则会按照配置顺序的反序执?。
自定义拦截器
实现HandlerIntercetor接口
/**
* ?定义springmvc拦截器
*/
public class MyIntercepter01 implements HandlerInterceptor {
/**
* 会在handler?法业务逻辑执?之前执?
注册SpringMVC拦截器
* 往往在这?完成权限校验?作
* @param request
* @param response
* @param handler
* @return 返回值boolean代表是否放?,true代表放?,false代表中?
* @throws Exception
*/
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse
response, Object handler) throws Exception {
//业务逻辑例如权限校验
System.out.println("MyIntercepter01 preHandle......");
return true;
}
/**
* 会在handler?法业务逻辑执?之后尚未跳转??时执?
* @param request
* @param response
* @param handler
* @param modelAndView 封装了视图和数据,此时尚未跳转??呢,你可以在这?针对返回的
数据和视图信息进?修改
* @throws Exception
*/
@Override
public void postHandle(HttpServletRequest request, HttpServletResponse
response, Object handler, ModelAndView modelAndView) throws Exception {
System.out.println("MyIntercepter01 postHandle......");
}
/**
* ??已经跳转渲染完毕之后执?
* @param request
* @param response
* @param handler
* @param ex 可以在这?捕获异常
* @throws Exception
*/
@Override
public void afterCompletion(HttpServletRequest request,
HttpServletResponse response, Object handler, Exception ex) throws Exception {
System.out.println("MyIntercepter01 afterCompletion......");
}
}
注册SpringMVC拦截器
<mvc:interceptors>
<!--拦截所有handler-->
<!--<bean class="com.lagou.edu.interceptor.MyIntercepter01"/>-->
<mvc:interceptor>
<!--配置当前拦截器的url拦截规则,**代表当前?录下及其??录下的所有url-->
<mvc:mapping path="/**"/>
<!--exclude-mapping可以在mapping的基础上排除?些url拦截-->
<!--<mvc:exclude-mapping path="/demo/**"/>-->
<bean class="com.lagou.edu.interceptor.MyIntercepter01"/>
</mvc:interceptor>
<mvc:interceptor>
<mvc:mapping path="/**"/>
<bean class="com.lagou.edu.interceptor.MyIntercepter02"/>
</mvc:interceptor>
</mvc:interceptors>
所需jar包
<!--?件上传所需jar坐标-->
<dependency>
<groupId>commons-fileupload</groupId>
<artifactId>commons-fileupload</artifactId>
<version>1.3.1</version>
</dependency>
配置文件解析器
<!--配置?件上传解析器,id是固定的multipartResolver-->
<bean id="multipartResolver"
class="org.springframework.web.multipart.commons.CommonsMultipartResolver">
<!--设置上传??,单位字节-->
<property name="maxUploadSize" value="1000000000"/>
</bean>
前端Form
<%--
1 method="post"
2 enctype="multipart/form-data"
3 type="file"
--%>
<form method="post" enctype="multipart/form-data" action="/demo/upload">
<input type="file" name="uploadFile"/>
<input type="submit" value="上传"/>
</form>
后台接收Handler
@RequestMapping("upload")
public String upload(MultipartFile uploadFile, HttpServletRequest request)
throws IOException {
// ?件原名,如xxx.jpg
String originalFilename = uploadFile.getOriginalFilename();
// 获取?件的扩展名,如jpg
String extendName =originalFilename.substring(originalFilename.lastIndexOf(".") + 1,originalFilename.length());
String uuid = UUID.randomUUID().toString();
// 新的?件名字
String newName = uuid + "." + extendName;
String realPath =request.getSession().getServletContext().getRealPath("/uploads");
// 解决?件夹存放?件数量限制,按?期存放
String datePath = new SimpleDateFormat("yyyy-MM-dd").format(new Date());
File floder = new File(realPath + "/" + datePath);
if(!floder.exists()) {
floder.mkdirs();
}
uploadFile.transferTo(new File(floder,newName));
return "success";
}
public class GlobalExceptionResolver {
//接受的异常的类型,也可以写Exception捕获所有异常都会返回到error页面
@ExceptionHandler(ArithmeticException.class)
public ModelAndView handleException(ArithmeticException exception,
HttpServletResponse response) {
ModelAndView modelAndView = new ModelAndView();
modelAndView.addObject("msg",exception.getMessage());
modelAndView.setViewName("error");
return modelAndView;
}
}
原文:https://www.cnblogs.com/jingzihao/p/14950253.html