发表于: 2021-11-12 20:21:57

1 897


学习Spring MVC 执行流程

spring MVC 执行流程

执行流程

1、用户发送请求到前端控制器DispatcherServlet

2、DispatcherServlet收到请求调用处理映射器HandlerMapping

3、处理映射器根据请求url找到具体的处理器,生成处理器执行链HandlerExecutionChain(包含处理器对象和处理器拦截器)返回给DispatcherServlet

4、DispatcherServlet根据处理器Handler获取对应的适配器

5、HandlerAdapter调用处理器Handler

6、Handler(Controller)执行完成后返回ModelAndView

7、HandlerAdapter返回ModelAndView

8、DispatcherServlet统一将返回的ModelAndView派送到ViewResolve(视图解析器)解析

9,视图解析器解析之后返回View

10、对View进行渲染

11、响应用户

调用流程

DispatcherServlet中使用的Bean

DispatcherServlet默认使用WebApplicationContext作为上下文,因此我们来看一下该上下文中有哪些特殊的Bean:

1、Controller:处理器/页面控制器,做的是MVC中的C的事情,但控制逻辑转移到前端控制器了,用于对请求进行处理;

2、HandlerMapping:请求到处理器的映射,如果映射成功返回一个HandlerExecutionChain对象(包含一个Handler处理器(页面控制器)对象、多个HandlerInterceptor拦截器)对象;如BeanNameUrlHandlerMapping将URL与Bean名字映射,映射成功的Bean就是此处的处理器;

3、HandlerAdapter:HandlerAdapter将会把处理器包装为适配器,从而支持多种类型的处理器,即适配器设计模式的应用,从而很容易支持很多类型的处理器;如SimpleControllerHandlerAdapter将对实现了Controller接口的Bean进行适配,并且掉处理器的handleRequest方法进行功能处理;

4、ViewResolver:ViewResolver将把逻辑视图名解析为具体的View,通过这种策略模式,很容易更换其他视图技术;如InternalResourceViewResolver将逻辑视图名映射为jsp视图;

5、LocalResover:本地化解析,因为Spring支持国际化,因此LocalResover解析客户端的Locale信息从而方便进行国际化;

6、ThemeResovler:主题解析,通过它来实现一个页面多套风格,即常见的类似于软件皮肤效果;

7、MultipartResolver:文件上传解析,用于支持文件上传;

8、HandlerExceptionResolver:处理器异常解析,可以将异常映射到相应的统一错误界面,从而显示用户友好的界面(而不是给用户看到具体的错误信息);

9、RequestToViewNameTranslator:当处理器没有返回逻辑视图名等相关信息时,自动将请求URL映射为逻辑视图名;

10、FlashMapManager:用于管理FlashMap的策略接口,FlashMap用于存储一个请求的输出,当进入另一个请求时作为该请求的输入,通常用于重定向场景,后边会细述。


RESTful风格阐述

REST服务是一种ROA(Resource-Oriented Architecture,面向资源的架构)应用。主要特点是方法信息存在于HTTP协议的方法中(GET,POST,PUT,DELETE),作用域存在于URL中。例如,在一个获取设备资源列表的GET请求中,方法信息是GET,作用域信息是URI种包含的对设备资源的过滤、分页和排序等条件
良好的REST API不需要任何文档
1.1REST风格资源路径
REST风格的资源路径设计是面向资源的,资源的名称应该是准确描述该资源的名词。
资源路径概览:sheme://host:port/path?queryString
例:http://localhost:8080/bywlstud
1.2HTTP方法
GET用于读取、检索、查询、过滤资源
PSOT用于创建一个资源
PUT用于修改、更新资源、创建客户端维护主键信息的资源
DELETE用于删除资源
资源地址和HTTP方法结合在一起就可以实现对资源的完整定位

1.3RESTful风格API设计
上文讲述了通过HTTP方法和资源路径对服务器的一个资源进行定位的过程
接下来看一个REST风格API的设计

可以看到通过这个RESTAPI都是通过对同一个资源的操作,所不同的就是通过不同的HTTP方法来实现对资源不同的处理。
2.MVC对REST的支持
1.1主要通过注解来实现

  • @Controller声名一个处理请求的控制器
  • @RequestMapping请求映射地址,它存在几个子注解对于实现REST风格来说更加具有语义性
    • GETMapping GET请求
    • PUTMapping PUT请求
    • POSTMapping POST请求
    • DELETEMapping DELETE请求sponseBody 将响应内容转换为JSON格式
  • @RequestBody 请求内容转换为JSON格式
  • @PathVariable("id")用于绑定一个参数
  • @RESTController 等同于@Controller+@ResponseBody在类上写了这个注解,标识这个类的所有方法只返回数据,而不进行视图跳转

1.2返回HTTP状态码
REST风格API一个最鲜明的特点通过返回对应的HTTPStatus来判断客户端的操作是否完成
下面是spring中关于Http状态码描述的枚举类,本文列举了常见的状态码(读者若对此感兴趣可以查看HttpStatus源码)
 public enum HttpStatus{
     OK(200, "OK"),//用于服务器有实体响应
     CREATED(201, "Created"),//创建了新实体,响应该实体
     NO_CONTENT(204, "No Content"),//服务器正常响应,但无实体响应
     BAD_REQUEST(400, "Bad Request"),//客户端请求语法错误
     NOT_FOUND(404, "Not Found"),//目标资源不存在
     INTERNAL_SERVER_ERROR(500, "Internal Server Error"),//服务器内部错误
     NOT_IMPLEMENTED(501, "Not Implemented"),//服务器不支持当前请求
 }
Spring返回状态码是通过@ResponseStatus注解或者ResponseEntity<?>类实现的。
@ResponseStatus方式
 @GetMapping(path = "/user/{id}" , produces = "application/json;charset=utf-8")
 @ResponseStatus(HttpStatus.OK)
 public User findUserById(@PathVariable("id")Integer id){
     User user = userService.findUserById(id);
     return user ;
 }
ResponseEntity<?>方式
 @GetMapping(produces = "application/json;charset=utf-8")
 public ResponseEntity<List<User>> findAll(){
     List<User]] users = userService.findAll();
     return new ResponseEntity<List<User>>(users , HttpStatus.OK);
 }
1.3由于MVC默认不支持PUT和DELETE方法,所以需要手动开启
在tomcat服务器的web.xml文件中开启一下配置
 <servlet>
         <servlet-name>default</servlet-name>
         <servlet-class>org.apache.catalina.servlets.DefaultServlet</servlet-class>
         <init-param>
             <param-name>debug</param-name>
             <param-value>0</param-value>
         </init-param>
         <init-param>
             <param-name>listings</param-name>
             <param-value>false</param-value>
         </init-param>
         <init-param>
         <param-name>readonly</param-name>
         <param-value>true</param-value><!--开启这个-->
         </init-param>
         <load-on-startup>1</load-on-startup>
     </servlet>
在项目的web.xml中配置
 <filter>
     <filter-name>HiddenHttpMethodFilter</filter-name>
     <filter-class>org.springframework.web.filter.HiddenHttpMethodFilter</filter-class>
   </filter>
 
   <filter-mapping>
     <filter-name>HiddenHttpMethodFilter</filter-name>
     <servlet-name>dispathcherServlet</servlet-name>
   </filter-mapping>
3.MVC实现REST代码实现

可以看到通过这个RESTAPI都是通过对同一个资源的操作,所不同的就是通过不同的HTTP方法来实现对资源不同的处理。
2.MVC对REST的支持
1.1主要通过注解来实现

  • @Controller声名一个处理请求的控制器
  • @RequestMapping请求映射地址,它存在几个子注解对于实现REST风格来说更加具有语义性
    • GETMapping GET请求
    • PUTMapping PUT请求
    • POSTMapping POST请求
    • DELETEMapping DELETE请求sponseBody 将响应内容转换为JSON格式
  • @RequestBody 请求内容转换为JSON格式
  • @PathVariable("id")用于绑定一个参数
  • @RESTController 等同于@Controller+@ResponseBody在类上写了这个注解,标识这个类的所有方法只返回数据,而不进行视图跳转

1.2返回HTTP状态码
REST风格API一个最鲜明的特点通过返回对应的HTTPStatus来判断客户端的操作是否完成
下面是spring中关于Http状态码描述的枚举类,本文列举了常见的状态码(读者若对此感兴趣可以查看HttpStatus源码)
 public enum HttpStatus{
     OK(200, "OK"),//用于服务器有实体响应
     CREATED(201, "Created"),//创建了新实体,响应该实体
     NO_CONTENT(204, "No Content"),//服务器正常响应,但无实体响应
     BAD_REQUEST(400, "Bad Request"),//客户端请求语法错误
     NOT_FOUND(404, "Not Found"),//目标资源不存在
     INTERNAL_SERVER_ERROR(500, "Internal Server Error"),//服务器内部错误
     NOT_IMPLEMENTED(501, "Not Implemented"),//服务器不支持当前请求
 }
Spring返回状态码是通过@ResponseStatus注解或者ResponseEntity<?>类实现的。
@ResponseStatus方式
 @GetMapping(path = "/user/{id}" , produces = "application/json;charset=utf-8")
 @ResponseStatus(HttpStatus.OK)
 public User findUserById(@PathVariable("id")Integer id){
     User user = userService.findUserById(id);
     return user ;
 }
ResponseEntity<?>方式
 @GetMapping(produces = "application/json;charset=utf-8")
 public ResponseEntity<List<User>> findAll(){
     List<User]] users = userService.findAll();
     return new ResponseEntity<List<User>>(users , HttpStatus.OK);
 }
1.3由于MVC默认不支持PUT和DELETE方法,所以需要手动开启
在tomcat服务器的web.xml文件中开启一下配置
 <servlet>
         <servlet-name>default</servlet-name>
         <servlet-class>org.apache.catalina.servlets.DefaultServlet</servlet-class>
         <init-param>
             <param-name>debug</param-name>
             <param-value>0</param-value>
         </init-param>
         <init-param>
             <param-name>listings</param-name>
             <param-value>false</param-value>
         </init-param>
         <init-param>
         <param-name>readonly</param-name>
         <param-value>true</param-value><!--开启这个-->
         </init-param>
         <load-on-startup>1</load-on-startup>
     </servlet>
在项目的web.xml中配置
 <filter>
     <filter-name>HiddenHttpMethodFilter</filter-name>
     <filter-class>org.springframework.web.filter.HiddenHttpMethodFilter</filter-class>
   </filter>
 
   <filter-mapping>
     <filter-name>HiddenHttpMethodFilter</filter-name>
     <servlet-name>dispathcherServlet</servlet-name>
   </filter-mapping>

3.3控制层代码
 @RestController
 @RequestMapping("/users")
 public class UserControler {
 
     @Autowired
     private IUserService userService ;
 
     //REST风格实现方法
 
     /**
      * 查询所有
      * @return
      */
     @GetMapping(produces = "application/json;charset=utf-8")
     public ResponseEntity<List<User>> findAll(){
         List<User]] users = userService.findAll();
         return new ResponseEntity<List<User>>(users , HttpStatus.OK);
     }
 
     /**、
      * 根据ID查询
      * @param id
      * @return
      */
 
     @GetMapping(path = "/{id}" , produces = "application/json;charset=utf-8")
     @ResponseStatus(HttpStatus.OK)
     public User findUserById(@PathVariable("id")Integer id){
         User user = userService.findUserById(id);
         return user ;
     }
     /**
      * 增加一个用户
      * 返回该用户
      */
     @PostMapping(produces = "application/json;charset=utf-8")
     @ResponseStatus(HttpStatus.CREATED)
     public User addUser(@RequestBody User user){
         User newUser = userService.addUser(user);
         return newUser ;
     }
 
     /**
      * 更新
      * @param user
      */
     @PutMapping(path = "/{id}" ,produces = "application/json;charset=utf-8")
     public ResponseEntity<User]] updateUser(@PathVariable("id") Integer id , @RequestBody User user){
         user.setUid(id);
         //资源是否修改
         boolean flag = userService.updateUser(user);
         User deUser = userService.findUserById(id);
         if(flag)
             return new ResponseEntity<User>(deUser,HttpStatus.CREATED);
         return new ResponseEntity<User>(deUser,HttpStatus.OK);
     }
 
     @DeleteMapping(path = "/{id}"  , produces = "application/json;charset=utf-8")
     @ResponseStatus(HttpStatus.NO_CONTENT)
     public void delUser(@PathVariable("id") Integer id){
         User user = userService.findUserById(id);
         userService.delUser(id);
     } 
}



返回列表 返回列表
评论

    分享到