概念

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

异常类

  定义一个全局的异常类,有异常信息,都交到这边来。它像一个污水处理厂,汇集所有的工业污水,然后分门别类进行污水净化。要现实这种功能就要用到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. jmeter安装与部署

    1.JDK组件安装 jmeter是一款java开发的测试软件,需要JDK运行环境支持,因此在安装jmeter前需要安装调试好jdk运行环境.jmeter3.2需要1.8以上版本jdk才可以正常运行.首 ...

  2. Node.js Windows Example

    Firstly, download the msi file from https://nodejs.org/en/ Second, click the msi file to install nod ...

  3. zabbix报警信息设置

    报警信息 默认标题: 服务器监控报警 告警主机:{HOST.NAME} 主机IP: {HOST.IP} 告警级别:{TRIGGER.SEVERITY} 告警信息:{TRIGGER.NAME} 问题详情 ...

  4. 8天入门docker系列 —— 第七天 让你的container实现跨主机访问

    当你有若干个容器之后,你可能就希望实现容器的跨机部署访问了,比如aspnetcore在一台host上,mysql在另外一个host上,如果要实现这样的功能,需要你借助 docker自带的overlay ...

  5. centos7搭建基于SAMBA的网络存储

    学习目标: 通过本实验理解Linux系统下SAMBA服务器和客户端的配置,实现客户机可自动挂载服务端的共享存储. 操作步骤: 1.  SAMBA服务器搭建 2.  SAMBA客户端配置 参考命令:   ...

  6. Quartz每次调度时被执行两次

    [关键字:重复执行.重复调用.每次执行两次.执行2次] 前言: 先说一下,项目背景.由于组内某成员在用Maven搭建项目时不规范,导致项目的名称与实际访问项目名称不一致.在部署项目时,必需要配一下虚拟 ...

  7. POJ 3621:Sightseeing Cows(最优比率环)

    http://poj.org/problem?id=3621 题意:有n个点m条有向边,每个点有一个点权val[i],边有边权w(i, j).找一个环使得Σ(val) / Σ(w)最大,并输出. 思路 ...

  8. 【无线安全实践入门】网络扫描和ARP欺骗

    文中可能存在错误操作或错误理解,望大家不吝指正. 同时也希望可以帮助到想要学习接触此方面.或兴趣使然的你,让你有个大概的印象. !阅前须知! 本文是基于我几年前的一本笔记本,上面记录了我学习网络基础时 ...

  9. 如何实现LRU算法?

    1.什么是LRU算法? LRU是一种缓存淘汰机制策略. 计算机的缓存容量有限,如果缓存满了就要删除一些内容,给新的内容腾位置.但是要删除哪些内容呢?我们肯定希望删掉那些没有用的缓存,而把有用的数据继续 ...

  10. 自己实现定制自己的专属java锁,来高效规避不稳定的第三方

    java juc 包下面已经提供了很多并发锁工具供我们使用,但在日常开发中,为了各种原因我们总是会用多线程来并发处理一些问题,然而并不是所有的场景都可以使用juc 或者java本身提供的锁来方便的帮助 ...