概念

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

异常类

  定义一个全局的异常类,有异常信息,都交到这边来。它像一个污水处理厂,汇集所有的工业污水,然后分门别类进行污水净化。要现实这种功能就要用到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. java文件操作和文件过滤器

    文件操作 java提供了File类来供我们操作文件,FIle是文件或文件夹的抽象表示,也就是对象表示.总结了几个常用的方法 File(File parenrt,String child).FIle(S ...

  2. 基于STM32F429和Cube的主从定时器多通道输出固定个数的PWM波形

    主从定时器的原理已在上篇博文: 基于STM32F429+HAL库编写的定时器主从门控模式级联输出固定个数PWM脉冲的程序 讲解了,这篇重点就讲如何实现多通道的PWM级联输出. 1.软件环境 Keil5 ...

  3. 【python3两小时快速入门】入门笔记01:基础

    又要我搞爬虫了,这次的源网站使用的ajax加载数据,我用java爬下来的页面内容部分全都是空,虽然java也有插件,但是使用起来感觉很麻烦,所以,python!老子来了.    1. 版本:pytho ...

  4. Educational Codeforces Round 66 (Rated for Div. 2) A

    A. From Hero to Zero 题目链接:http://codeforces.com/contest/1175/problem/A 题目 ou are given an integer n ...

  5. Visual Studio模板代码注释小技巧分享

    在日常开发过程中,难免有这样一种需求:就是你所建的每一个类文件或者接口文件都需要标注下作者姓名以及类的用途.如果我们每次创建文件的时候都需要写一遍这些信息是很烦神的.还好Visual Studio给我 ...

  6. HTTP 学习笔记01

    HTTP   hypertext transfer protocol (超文本传输协议) TCP/IP 协议集中的一个应用层协议 用于定义WEB浏览器与WEB服务器之间交换数据的过程以及数据本身的格式 ...

  7. .NET中生成动态验证码

    .NET中生成动态验证码 验证码是图片上写上几个字,然后对这几个字做特殊处理,如扭曲.旋转.修改文字位置,然后加入一些线条,或加入一些特殊效果,使这些在人类能正常识别的同时,机器却很难识别出来,以达到 ...

  8. redis 发布和订阅实现

    参考文献 15天玩转redis -- 第九篇 发布/订阅模式 <Redis设计与实现> 命令简介 在redis用户手册中,跟发布订阅相关的命令有如下的六个: PSUBSCRIBE PUBL ...

  9. flink window实例分析

    window是处理数据的核心.按需选择你需要的窗口类型后,它会将传入的原始数据流切分成多个buckets,所有计算都在window中进行. flink本身提供的实例程序TopSpeedWindowin ...

  10. PostgreSQL 窗口函数 ( Window Functions ) 如何使用?

    一.为什么要有窗口函数 我们直接用例子来说明,这里有一张学生考试成绩表testScore: 现在有个需求,需要查询的时候多出一列subject_avg_score,为此科目所有人的平均成绩,好跟每个人 ...