SpringMVC进阶

springmvc进阶

一 ajax异步交互

  • 将JSON格式的数据进行转换

  • Springmvc默认用MappingJackson2HttpMessageConverter对json数据进行转换,需要加入 jackson的包;同时使用<mvc:annotation-driven />(处理器映射器、处理器适配器增强)。

  • 相关依赖:

      <!--ajax异步交互-->
            <dependency>
                <groupId>com.fasterxml.jackson.core</groupId>
                <artifactId>jackson-databind</artifactId>
                <version>2.9.8</version>
            </dependency>
            <dependency>
                <groupId>com.fasterxml.jackson.core</groupId>
                <artifactId>jackson-core</artifactId>
                <version>2.9.8</version>
            </dependency>
            <dependency>
                <groupId>com.fasterxml.jackson.core</groupId>
                <artifactId>jackson-annotations</artifactId>
                <version>2.9.0</version>
            </dependency>
    

1.1 @RequestBody

  • 在形参指定要转换的JSON字段。

  • jsp中:<script>标签不能自闭合!!

    <body>
    <%-- ajax异步交互 --%>
    <script src="${pageContext.request.contextPath}/js/jquery-3.4.1.min.js">
      </script>
    <button id="btn1">ajax异步提交</button>
    <script>
        $("#btn1").click(function () {
            let url = '${pageContext.request.contextPath}/day2/ajaxRequest';
            let data = '[{"id":1,"username":"张三"},{"id":2,"username":"李四"}]';
            $.ajax({
                type: 'post',
                url: url,
                data: data,
                contentType: 'application/json;charset=utf-8',
                success: function (resp) {
                    alert(JSON.stringify(resp))
                }
            })
        })
    </script>
    </body>
    

1.2 @ResponseBody

  • 用于将Controller的方法返回的对象,转换成JSON等格式,通过response返回给客户端。

1.3 Controller

@Controller
@RequestMapping("/day2")//一级目录
public class AjaxController {
/*
ajax异步交互
@RequestBody:将JSON转为Pojo对象
@ResponseBody:转换为JSON串
 */
    @RequestMapping("/ajaxRequest")
    @ResponseBody
    public List<User> ajaxRequest(@RequestBody List<User> list) {
        System.out.println(list);
        //[User{id=1, username='张三'}, User{id=2, username='李四'}]
        return list;
    }
}

二 RESTful

2.1 什么是RESTful

  • Restful是一种软件架构风格、设计风格,而不是标准,只是提供了一组设计原则和约束条件。主要用于客户端和服务器交互类的软件,基于这个风格设计的软件可以更简洁,更有层次,更易于实现缓存机制等。
  • Restful风格的请求是使用“url+请求方式”表示一次请求目的的,HTTP 协议里面四个表示操作方式的动词如下:
    • GET:读取(Read)
    • POST:新建(Create)
    • PUT:更新(Update)
    • DELETE:删除(Delete)
    • 可以使用postman测试不同请求方式
客户端请求原来风格URL地址RESTful风格URL地址
查询所有/user/findAllGET /user
根据ID查询/user/findById?id=1GET /user/{1}
新增/user/savePOST /user
修改/user/updatePUT /user
删除/user/delete?id=1DELETE /user/{1}

2.2 代码实现

1. @PathVariable
  • 接收RESTful风格请求地址中占位符的值
2. @RestController
  • 前后端分离项目中,前端通过Ajax异步交互,后端直接返回JSON格式数据(代替@ResponseBody),和@Controller。

@RequestMapping("/restful")
@RestController//代替:@Controller+@ResponseBody(返回JSON格式)
public class RestfulController {

    //查询所有
    @GetMapping("/user")
    public String get(){
        return "get";
    }

    //根据id查询
    /*
    @PathVariable接收占位符中的值
     */
    @GetMapping("/user/{id}")
    public String get(@PathVariable Integer id){
        return "get:"+id;
    }

    //新增
    @PostMapping("/user")
    public String post(){
        return "post";
    }

    //更新
    @PutMapping("/user")
    public String put(){
        return "put";
    }

    //删除
    @DeleteMapping("/user/{id}")
    public String delete(@PathVariable Integer id){
        return "delete"+id;
    }
}

三 文件上传

3.1 文件上传三要素

  1. input属性为file类型
  2. 提交方式post
  3. 表单的enctype属性是多部分表单形式 enctype=“multipart/form-data"

3.2 文件上传

1. 导入fileupload和io坐标
        <dependency>
            <groupId>commons-fileupload</groupId>
            <artifactId>commons-fileupload</artifactId>
            <version>1.3.3</version>
        </dependency>
        <dependency>
            <groupId>commons-io</groupId>
            <artifactId>commons-io</artifactId>
            <version>2.6</version>
        </dependency>
2. 配置文件上传解析器
  • spring-mvc.xml
    <!--配置文件上传解析器 id必须为multipartResolver!!!-->
    <bean class="org.springframework.web.multipart.commons.CommonsMultipartResolver"
          id="multipartResolver">
        <!--设置文件最大值5M-->
        <property name="maxUploadSize" value="5242880"/>
        <!-- 设定文件上传时写入内存的最大值,如果小于这个参数不会生成临时文件,默认为10240 -->
        <property name="maxInMemorySize" value="40960"></property>
    </bean>
3. jsp页面
<form action="${pageContext.request.contextPath}/fileUpload" method="post"
      enctype="multipart/form-data">
    名称:<input type="text" name="username"> <br>
    文件:<input type="file" name="filePic"> <br>
    <input type="submit" value="单文件上传">
</form>
4.fileUploadController
@Controller
public class FileUploadController {
    /*
    单文件上传
    MultipartFile:文件类型
     */
    @RequestMapping("/fileUpload")
    public String fileUpload(String username, MultipartFile filePic) throws
            IOException {
        System.out.println(username);
		//表单中的name名
        System.out.println(filePic.getName());
        //获取文件名
        String originalFilename = filePic.getOriginalFilename();
		//保存文件
        filePic.transferTo(new File("C:/test/" + originalFilename));
        return "success";
    }
}
  • 多文件上传只需将jsp中的name相同,后台使用数组接收,遍历存入。
public String filesUpload(String username, MultipartFile[] filePic) 
 for (MultipartFile multipartFile : filePic) {……}  

四 异常处理

4.1 异常处理的思路

  • 在java中一般有两种异常处理方式
    1. 一种是当前方法捕获处理(try-catch),这种处理方式会造成业务代码和异常处理代码的耦合。
    2. 层层上抛进行处理,在此基础上衍生出了SpringMVC的异常处理机制。
  • 系统的dao、service、controller出现都通过throws Exception向上抛出,最后由springmvc前端控制器交由异常处理器进行异常处理,如下图:

4.2 自定义异常处理器

1.创建异常处理器类实现HandlerExceptionResolver
2.配置异常处理器
@Component//交给Spring容器
public class GlobalExceptionResolver implements HandlerExceptionResolver {
     /*
      Exception e:产生的具体异常对象
      */
    @Override
    public ModelAndView resolveException(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse, Object o, Exception e) {
       ModelAndView modelAndView=new ModelAndView();
       modelAndView.addObject("error",e.getMessage());//异常类型
       modelAndView.setViewName("error");//跳转页面
        return modelAndView;
    }
}

3.异常页面
<body>
${error}
</body>
4. 测试异常跳转
@Controller
public class ExceptionController {
    @RequestMapping("/testException")
    public String testException() {
        int i = 1 / 0;
        return "success";
    }
}

4.3 web的处理异常机制

  • 我们对404 500等异常进行处理。
    <!--处理404 500异常-->
    <error-page>
        <error-code>500</error-code>
        <location>/500.jsp</location>
    </error-page>
    <error-page>
        <error-code>404</error-code>
        <location>/404.jsp</location>
    </error-page>
  • 配置后会跳转到我们配置的具体页面。

五 拦截器

5.1 拦截器(interceptor)的作用

  • Spring MVC的拦截器类似于 Servlet 开发中的过滤器 Filter,用于对处理器进行预处理和后处理。对于多个拦截器,按配置的先后执行。

5.2 拦截器和过滤器区别

  • 从作用范围(MVC框架专用)与拦截范围(只拦截访问控制器方法)解析

5.3 代码实现

1.创建拦截器类实现HandlerInterceptor接口
public class MyInterceptor1 implements HandlerInterceptor {

    //执行前拦截
    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
        System.out.println("preHandle1");
        return true; //false:不放行
    }

    //目标方法执行后.视图对象返回前(return前)
    @Override
    public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
        System.out.println("postHandle1");
    }

    //所有执行后,执行(return后)
    @Override
    public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
        System.out.println("afterCompletion1");
    }
}

2.配置拦截器
    <mvc:interceptors>
        <mvc:interceptor>
            <!--**对所有controller方法拦截-->
            <mvc:mapping path="/**"/>
            <bean class="com.zz.interceptor.MyInterceptor1"/>
        </mvc:interceptor>
    </mvc:interceptors>
3.测试拦截器的拦截效果
  • 跳转后的jsp页面:

    <body>
        <% System.out.println("视图执行了....");%>
    </body>
    
  • Controller

    @Controller
    public class TargetController {
        @RequestMapping("/target")
        public String targetMethod() {
            System.out.println("目标方法执行了...");
            return "success";
            /*
            preHandle1
            目标方法执行了...
            postHandle1
            视图执行了....
            afterCompletion1
             */
        }
    }
    

5.4 拦截器链

  • 开发中拦截器可以单独使用,也可以同时使用多个拦截器形成一条拦截器链。开发步骤和单个拦截器是一样的,只不过注册的时候注册多个,注意这里注册的顺序就代表拦截器执行的顺序。

  • 只有当前拦截器的preHandle返回true,当前拦截器的后续方法才调用

  • 示例:

     /*
      preHandle1
      目标方法执行了...
      postHandle1
      视图执行了....
      afterCompletion1
       */
  }

}


### 5.4 拦截器链

+ 开发中拦截器可以单独使用,也可以同时使用多个拦截器形成一条拦截器链。开发步骤和单个拦截器是一样的,只不过注册的时候注册多个,注意这里注册的顺序就代表拦截器执行的顺序。

+ 只有当前拦截器的preHandle返回true,当前拦截器的后续方法才调用

[外链图片转存中...(img-wqRcc9pA-1615978445642)]

+ 示例:

[外链图片转存中...(img-r1TsGmdL-1615978445644)]

+ 执行顺序:方法执行前1->方法执行前2(flase)->页面渲染后1
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值