概念

  异常,在程序中经常发生,如果发生异常怎样给用户一个良好的反馈体验就是我们需要处理的问题。以前处理异常信息,经常都是给前端一个统一的响应,如数据错误,程序崩溃等等。没办法指出哪里出错了,这是一种对用户很不友好的体验。我们应该根据自己的业务给予信息提示

异常类

  定义一个全局的异常类,有异常信息,都交到这边来。它像一个污水处理厂,汇集所有的工业污水,然后分门别类进行污水净化。要现实这种功能就要用到springBoot的@ControllerAdvice注解,它的作用是控制器增加,应用到有以下的注解的函数或类@ExceptionHandler,@InitBinder,@ModelAttribute。

创建一个异常类

package com.xmlxy.exception;

import org.springframework.web.bind.annotation.ControllerAdvice;
import org.springframework.web.bind.annotation.ExceptionHandler;
import org.springframework.web.bind.annotation.ResponseBody; import java.util.HashMap;
import java.util.Map; @ControllerAdvice
public class CommonExceptionHandler { @ExceptionHandler(Exception.class)
@ResponseBody
public Map<String,Object> exceptionHandler(Exception e)
{
Map<String,Object> result = new HashMap<String,Object>();
result.put("code","9999");
result.put("msg","异常被捕获了");
return result;
}
}

写个接口测试一下

@RequestMapping(value = "test",method = RequestMethod.GET)
public String test()
{
int i = 1 / 0;
return "test接口";
}

我们都知道这会让控制台抛出一个除数不能为零的异常信息。调用接口,会发现控制台并没有异常信息,页面输出

{"msg":"除数不能为零","code":"9999"}

然后前端根据我们约定的code,或弹出窗,或跳转页面。但是这样是远远不够的,因为异常信息很多,我上文中,因为贪图省事,直接用Exception捕获并没有给予详细的异常捕捉。如果多余的不同异常,需要进行不同的异常处理,就可以编写多个exceptionHandler方法,可以指定处理异常类。

@ExceptionHandler(FileNotFoundException.class)
@ResponseBody
public Map<String,Object> exceptionHandler(FileNotFoundException e)
{
Map<String,Object> result = new HashMap<String, Object>();
result.put("code","8888");
result.put("msg","文件不存在");
return result;
}

自定义异常

  现有的异常有时并没有满足我们业务需求,就得自定义自己的专属异常类,举一个前几次讲的登录接口demo,用户可能没输密码直接点登录,是进行密码错误提示,还是反馈密码为空哪种体验比较好,很明显是后一种。

自定义异常,继承Exception接口

package com.xmlxy.exception;

public class FistSpringBootException extends Exception
{
private String code; private FistSpringBootException(String code)
{
super();
this.code = code;
} public FistSpringBootException(String code,String message)
{
super(message);
this.code = code;
} public String getCode() {
return code;
} public void setCode(String code) {
this.code = code;
} }

封装一个数据效验类

package com.xmlxy.exception;

import org.springframework.util.StringUtils;

public class ParamUtil
{
/*验证是否是数字*/
public static int isNumber(String args,String name) throws FistSpringBootException {
if (StringUtils.isEmpty(args) && !name.matches("^[0-9]*$"))
{
throw new FistSpringBootException("1111",name + "参数不合法");
}
return Integer.parseInt(args);
} /*验证参数是否为空*/
public static String isEmpty(String args,String name) throws FistSpringBootException {
if (StringUtils.isEmpty(args))
{
throw new FistSpringBootException("2222",name + "参数不能为空");
}
return String.valueOf(args);
}
}

登录接口

@RestController
public class LoginController { @RequestMapping(value = "login",method = RequestMethod.GET)
public String login(HttpServletRequest request) throws FistSpringBootException {
String user = ParamUtil.isEmpty(request.getParameter("user"),"user");
String pwd = ParamUtil.isEmpty(request.getParameter("pwd"),"pwd"); HttpSession session = request.getSession();
if ("admin".equals(user) && "admin".equals(pwd))
{
User user1 = new User();
user1.setUser(user);
user1.setPwd(pwd);
session.setAttribute("user",user1);
return "登录成功";
}
return "密码错误,登录失败";
}
}

进行异常捕获

package com.xmlxy.exception;

import org.springframework.web.bind.annotation.ControllerAdvice;
import org.springframework.web.bind.annotation.ExceptionHandler;
import org.springframework.web.bind.annotation.ResponseBody; import java.io.FileNotFoundException;
import java.util.HashMap;
import java.util.Map; @ControllerAdvice
public class CommonExceptionHandler { @ExceptionHandler(Exception.class)
@ResponseBody
public Map<String,Object> exceptionHandler(Exception e)
{
Map<String,Object> result = new HashMap<String,Object>();
result.put("code","9999");
result.put("msg","除数不能为零");
return result;
} @ExceptionHandler(FileNotFoundException.class)
@ResponseBody
public Map<String,Object> exceptionHandler(FileNotFoundException e)
{
Map<String,Object> result = new HashMap<String, Object>();
result.put("code","8888");
result.put("msg","文件不存在");
return result;
} @ExceptionHandler(FistSpringBootException.class)
@ResponseBody
public Map<String,Object> exceptionHandler(FistSpringBootException e)
{
Map<String,Object> result = new HashMap<String,Object>();
result.put("code","1111");
result.put("msg",e.getMessage());
return result;
} }

demo写完,直接测试一下,访问 http://127.0.0.1:8080/login?user=&pwd=

是否会发现自己还得效验数据有点费劲,没事,springBoot已经帮我们想好了,很简单,只要加上注解。

User类

package com.xmlxy.bean;

import lombok.Data;
import org.springframework.stereotype.Component; import javax.validation.constraints.NotBlank; @Component
@Data
public class User {
@NotBlank(message = "user不能为空")
private String user;
@NotBlank(message = "pwd不能为空")
private String pwd;
}

异常全局配置

    @ExceptionHandler(BindException.class)
@ResponseBody
public Map<String,Object> exceptionHandler(BindException e)
{
Map<String,Object> result = new HashMap<String,Object>();
result.put("code","1111");
result.put("msg","参数不合法");
return result;
}

登录接口

@RestController
public class LoginController { @RequestMapping(value = "login",method = RequestMethod.GET)
public String login(@Valid User user, HttpServletRequest request) {
try {
HttpSession session = request.getSession();
if ("admin".equals(user.getUser()) && "admin".equals(user.getPwd()))
{
session.setAttribute("user",user);
return "登录成功";
}
}catch (Exception e){
e.printStackTrace();
}
return "密码错误,登录失败";
}
}

再次访问,会发现已经帮你效验了

{"msg":"参数不合法","code":"1111"}

还有很多的数据效验注解,比如@Email注解,接收参数必须是email地址,还有限制长度的@Length注解,以下是常用的校验注解

@Null   被注释的元素必须为 null
@NotNull 被注释的元素必须不为 null
@AssertTrue 被注释的元素必须为 true
@AssertFalse 被注释的元素必须为 false
@Min(value) 被注释的元素必须是一个数字,其值必须大于等于指定的最小值
@Max(value) 被注释的元素必须是一个数字,其值必须小于等于指定的最大值
@DecimalMin(value) 被注释的元素必须是一个数字,其值必须大于等于指定的最小值
@DecimalMax(value) 被注释的元素必须是一个数字,其值必须小于等于指定的最大值
@Size(max=, min=) 被注释的元素的大小必须在指定的范围内
@Digits (integer, fraction) 被注释的元素必须是一个数字,其值必须在可接受的范围内
@Past 被注释的元素必须是一个过去的日期
@Future 被注释的元素必须是一个将来的日期
@Pattern(regex=,flag=) 被注释的元素必须符合指定的正则表达式
Hibernate Validator提供的校验注解:
@NotBlank(message =) 验证字符串非null,且长度必须大于0
@Email 被注释的元素必须是电子邮箱地址
@Length(min=,max=) 被注释的字符串的大小必须在指定的范围内
@NotEmpty 被注释的字符串的必须非空
@Range(min=,max=,message=) 被注释的元素必须在合适的范围内

springBoot数据校验与统一异常处理的更多相关文章

  1. SpringBoot小技巧:统一异常处理

    SpringBoot小技巧:统一异常处理 情景描述 对于接口的定义,我们通常会有一个固定的格式,比如: 但是调用方在请求我们的API时把接口地址写错了,就会得到一个404错误,且不同于我们定义的数据格 ...

  2. SpringBoot 2 快速整合 | 统一异常处理

    统一异常处理相关注解介绍 @ControllerAdvice 声明在类上用于指定该类为控制增强器类,如果想声明返回的结果为 RESTFull 风格的数据,需要在声明 @ExceptionHandler ...

  3. Springboot数据校验

    SpringBoot中使用了Hibernate-validate校验框架 1.在实体类中添加校验规则 校验规则: @NotBlank: 判断字符串是否为null或者是空串(去掉首尾空格).@NotEm ...

  4. SpringBoot入门系列(十一)统一异常处理的实现

    前面介绍了Spring Boot 如何整合定时任务已经Spring Boot 如何创建异步任务和定时任务.不清楚的朋友可以看看之前的文章:<Spring Boot 入门系列文章> 接下来主 ...

  5. SpringMVC第六篇【校验、统一处理异常】

    Validation 在我们的Struts2中,我们是继承ActionSupport来实现校验的-它有两种方式来实现校验的功能 手写代码 XML配置 这两种方式也是可以特定处理方法或者整个Action ...

  6. SpringBoot | 第八章:统一异常、数据校验处理

    前言 在web应用中,请求处理时,出现异常是非常常见的.所以当应用出现各类异常时,进行异常的捕获或者二次处理(比如sql异常正常是不能外抛)是非常必要的,比如在开发对外api服务时,约定了响应的参数格 ...

  7. springboot统一异常处理及返回数据的处理

    一.返回code数据的处理 代码: Result.java /** * http请求返回的最外层对象 * Created by 廖师兄 * 2017-01-21 13:34 */ public cla ...

  8. SpringBoot第七集:异常处理与整合JSR303校验(2020最新最易懂)

    SpringBoot第七集:异常处理与整合JSR303校验(2020最新最易懂) 一.SpringBoot全局异常 先讲下什么是全局异常处理器? 全局异常处理器就是把整个系统的异常统一自动处理,程序员 ...

  9. SpringBoot系列——自定义统一异常处理

    前言 springboot内置的/error错误页面并不一定适用我们的项目,这时候就需要进行自定义统一异常处理,本文记录springboot进行自定义统一异常处理. 1.使用@ControllerAd ...

随机推荐

  1. node.js简单数据接口开发

    随着网络时代的快速发展,前端开发不仅仅是做出漂亮的页面就可以了,还要会一点后端语言,那么后端语言有Java,php,node.js最常见,那我们应该学哪一种呢,为了让我们自己更好的学习,我推荐选择no ...

  2. iOS开发系列之性能优化(上)

    本篇主要记录一下我对界面优化上的一些探索.关于时间优化的探索将会在中篇里进行介绍.下篇将主要介绍一些耗电优化.安装包瘦身的探索. ### 1.卡顿原理 要了解卡顿原理,需要对帧缓冲区.垂直同步.CPU ...

  3. js深入(三)作用域链与闭包

    在之前我们根绝对象的原型说过了js的原型链,那么同样的js 万物皆对象,函数也同样存在这么一个链式的关系,就是函数的作用域链 作用域链 首先先来回顾一下之前讲到的原型链的寻找机制,就是实例会先从本身开 ...

  4. 简单的scrapy实例

    前天实验室的学长要求写一个简单的scrapy工程出来,之前也多少看了点scrapy的知识,但始终没有太明白,刚好趁着这个机会,加深一下对scrapy工作流程的理解.由于临近期末,很多作业要做(其实.. ...

  5. 浅入深出Vue:路由

    路由的概念在计算机界中的历史大概可以追溯到OSI模型中的数据链路层与网络层中的定义.这里的定义大意是:在转发数据包时,根据数据包的目的地址进行寻址,从而将数据包发往指定的目的地. 在 Web开发中同样 ...

  6. leadcode的Hot100系列--104. 二叉树的最大深度

    依然使用递归思想. 思路: 1.树的深度 = max (左子树深度,右子树深度)+ 1 . ------> 这里的加1是表示自己节点深度为1. 2.如果当前节点为null,则说明它的左右子树深度 ...

  7. ZooKeeper入门(三) ZooKeeper数据模型

    1 简述 ZooKeeper可以看成一种高可用性的文件系统,但是,它没有文件和目录,而是使用节点,称为znode. znode可以作为保存数据的容器(如同文件),也可以作为保存其他节点的容器(如同目录 ...

  8. concat的应用

    今天遇到一个问题,有一张车辆信息表,一张车辆品牌表,他们之间的品牌进行关联, 但是车辆信息表中品牌的名称较长,而品牌表名称较短.例如:车辆表:东风标致:品牌表:标致. 为了达到两种表的“模糊关联”. ...

  9. 为什么QQ能上却打不开网页呢?

    互联网是一个复杂又有趣的玩意儿,接下来我来分享一个案例: 网页打不开,但奇怪的是QQ可以正常使用??? 网页打不开的原因:电脑上Tcp/IP设置中没有使用自动分配的DNS服务器,而且自行设定的DNS服 ...

  10. web文件下载(附方案及源码配置)

    1. 场景描述 因项目需查询数据量比较大(需要查询Hbase等nosql数据库),采用用户点击查询后,后台查询并生成查询文件:然后消息通知用户后,用户点击下载的方式来满足用户需求. 2. 解决方案 W ...