Spring Boot 项目统一结果,统一异常,统一日志,写的太好了

Spring Boot 项目统一结果,统一异常,统一日志,写的太好了

统一结果返回

目前的前后端开发大部分数据的传输格式都是json,因此定义一个统一规范的数据格式有利于前后端的交互与UI的展示。

统一结果的一般形式

  • 是否响应成功;
  • 响应状态码;
  • 状态码描述;
  • 响应数据
  • 其他标识符
  • 结果类枚举

    • 前三者可定义结果枚举,如:success,code,message

    @Getterpublic enum ResultCodeEnum { SUCCESS(true,20000,”成功”), UNKNOWN_ERROR(false,20001,”未知错误”),, PARAM_ERROR(false,20002,”参数错误”), ; // 响应是否成功 private Boolean success; // 响应状态码 private Integer code; // 响应信息 private String message; ResultCodeEnum(boolean success, Integer code, String message) { this.success = success; this.code = code; this.message = message; }}

    统一结果类

    • 第5个属于自定义返回,利用前4者可定义统一返回对象

    注意:

  • 外接只可以调用统一返回类的方法,不可以直接创建,影刺构造器私有;
  • 内置静态方法,返回对象;
  • 为便于自定义统一结果的信息,建议使用链式编程,将返回对象设类本身,即return this;
  • 响应数据由于为json格式,可定义为JsonObject或Map形式;
  • @Datapublic class R { private Boolean success; private Integer code; private String message; private Map data = new HashMap(); // 构造器私有 private R(){} // 通用返回成功 public static R ok() { R r = new R(); r.setSuccess(ResultCodeEnum.SUCCESS.getSuccess()); r.setCode(ResultCodeEnum.SUCCESS.getCode()); r.setMessage(ResultCodeEnum.SUCCESS.getMessage()); return r; } // 通用返回失败,未知错误 public static R error() { R r = new R(); r.setSuccess(ResultCodeEnum.UNKNOWN_ERROR.getSuccess()); r.setCode(ResultCodeEnum.UNKNOWN_ERROR.getCode()); r.setMessage(ResultCodeEnum.UNKNOWN_ERROR.getMessage()); return r; } // 设置结果,形参为结果枚举 public static R setResult(ResultCodeEnum result) { R r = new R(); r.setSuccess(result.getSuccess()); r.setCode(result.getCode()); r.setMessage(result.getMessage()); return r; } /**————使用链式编程,返回类本身———–**/ // 自定义返回数据 public R data(Map map) { this.setData(map); return this; } // 通用设置data public R data(String key,Object value) { this.data.put(key, value); return this; } // 自定义状态信息 public R message(String message) { this.setMessage(message); return this; } // 自定义状态码 public R code(Integer code) { this.setCode(code); return this; } // 自定义返回结果 public R success(Boolean success) { this.setSuccess(success); return this; }}

    控制层返回

    • 视图层使用统一结果

    @RestController@RequestMapping(“/api/v1/users”)public class TeacherAdminController { @Autowired private UserService userService; @GetMapping public R list() { List list = teacherService.list(null); return R.ok().data(“itms”, list).message(“用户列表”); }}

    • json结果

    { “success”: true, “code”: 20000, “message”: “查询用户列表”, “data”: { “itms”: [ { “id”: “1”, “username”: “admin”, “role”: “ADMIN”, “deleted”: false, “gmtCreate”: “2019-12-26T15:32:29”, “gmtModified”: “2019-12-26T15:41:40” },{ “id”: “2”, “username”: “zhangsan”, “role”: “USER”, “deleted”: false, “gmtCreate”: “2019-12-26T15:32:29”, “gmtModified”: “2019-12-26T15:41:40” } ] }}

    统一结果类的使用参考了mybatis-plus中R对象的设计。


    统一异常处理

    使用统一返回结果时,还有一种情况,就是程序的保存是由于运行时异常导致的结果,有些异常我们可以无法提前预知,不能正常走到我们return的R对象返回。

    因此,我们需要定义一个统一的全局异常来捕获这些信息,并作为一种结果返回控制层

    @ControllerAdvice

    注解为统一异常处理的核心

    是一种作用于控制层的切面通知(Advice),该注解能够将通用的@ExceptionHandler、@InitBinder和@ModelAttributes方法收集到一个类型,并应用到所有控制器上

    该类中的设计思路:

  • 使用@ExceptionHandler注解捕获指定或自定义的异常;
  • 使用@ControllerAdvice集成@ExceptionHandler的方法到一个类中;
  • 必须定义一个通用的异常捕获方法,便于捕获未定义的异常信息;
  • 自定一个异常类,捕获针对项目或业务的异常;
  • 异常的对象信息补充到统一结果枚举中;
  • 自定义全局异常类

    @Datapublic class CMSException extends RuntimeException { private Integer code; public CMSException(Integer code, String message) { super(message); this.code = code; } public CMSException(ResultCodeEnum resultCodeEnum) { super(resultCodeEnum.getMessage()); this.code = resultCodeEnum.getCode(); } @Override public String toString() { return “CMSException{” + “code=” + code + “, message=” + this.getMessage() + ‘}’; }}

    统一异常处理器

    // …import org.springframework.web.bind.annotation.ControllerAdvice;import org.springframework.web.bind.annotation.ExceptionHandler;import org.springframework.web.bind.annotation.ResponseBody;@ControllerAdvicepublic class GlobalExceptionHandler { /**——– 通用异常处理方法 ——–**/ @ExceptionHandler(Exception.class) @ResponseBody public R error(Exception e) { e.printStackTrace(); return R.error(); // 通用异常结果 } /**——– 指定异常处理方法 ——–**/ @ExceptionHandler(NullPointerException.class) @ResponseBody public R error(NullPointerException e) { e.printStackTrace(); return R.setResult(ResultCodeEnum.NULL_POINT); } @ExceptionHandler(HttpClientErrorException.class) @ResponseBody public R error(IndexOutOfBoundsException e) { e.printStackTrace(); return R.setResult(ResultCodeEnum.HTTP_CLIENT_ERROR); } /**——– 自定义定异常处理方法 ——–**/ @ExceptionHandler(CMSException.class) @ResponseBody public R error(CMSException e) { e.printStackTrace(); return R.error().message(e.getMessage()).code(e.getCode()); }}

    控制层展示

    以下为展示当遇到null指定异常时,返回的结果信息

    { “success”: false, “code”: 20007, “message”: “空指针异常”, “data”: {}}


    统一日志收集

    日志是追踪错误定位问题的关键,尤其在生产环境中,需要及时修复热部署,不会提供开发者debug的环境,此时日志将会是最快解决问题的关键

    日志的框架比较丰富,由于spring boot对logback的集成,因此推荐使用logback在项目中使用。

    Logback

    配置

    以下直接贴出配置信息,介绍信息科直接参考备注

    <!– 日志级别从低到高分为TRACE < DEBUG < INFO < WARN < ERROR logback debug ${CONSOLE_LOG_PATTERN} UTF-8 ${log.path}/edu_debug.log %d{yyyy-MM-dd HH:mm:ss.SSS} [%thread] %-5level %logger{50} – %msg%n UTF-8 ${log.path}/web-debug-%d{yyyy-MM-dd}.%i.log 100MB 15 debug ACCEPT DENY ${log.path}/edu_info.log %d{yyyy-MM-dd HH:mm:ss.SSS} [%thread] %-5level %logger{50} – %msg%n UTF-8 ${log.path}/web-info-%d{yyyy-MM-dd}.%i.log 100MB 15 info ACCEPT DENY ${log.path}/edu_warn.log %d{yyyy-MM-dd HH:mm:ss.SSS} [%thread] %-5level %logger{50} – %msg%n UTF-8 ${log.path}/web-warn-%d{yyyy-MM-dd}.%i.log 100MB 15 warn ACCEPT DENY ${log.path}/edu_error.log %d{yyyy-MM-dd HH:mm:ss.SSS} [%thread] %-5level %logger{50} – %msg%n UTF-8 ${log.path}/web-error-%d{yyyy-MM-dd}.%i.log 100MB 15 ERROR ACCEPT DENY

    日志收集异常信息

    日志信息往往伴随着异常信息的输出,因此,我们需要修改统一异常的处理器,将异常信息以流的方式写到日志文件中

    • 异常信息文件工具类

    @Slf4jpublic class ExceptionUtil { /** * 打印异常信息 */ public static String getMessage(Exception e) { String swStr = null; try (StringWriter sw = new StringWriter(); PrintWriter pw = new PrintWriter(sw)) { e.printStackTrace(pw); pw.flush(); sw.flush(); swStr = sw.toString(); } catch (IOException ex) { ex.printStackTrace(); log.error(ex.getMessage()); } return swStr; }}

    • 修改统一异常处理器,将异常方法中的直接打印改为日志输入并打印

    // …import lombok.extern.slf4j.Slf4j;@ControllerAdvice@Slf4jpublic class GlobalExceptionHandler { /**——– 通用异常处理方法 ——–**/ @ExceptionHandler(Exception.class) @ResponseBody public R error(Exception e) { // e.printStackTrace(); log.error(ExceptionUtil.getMessage(e)); return R.error(); } // …}

    注意

  • 日志的环境即spring.profiles.acticve,跟随项目启动;
  • 启动后,即可到自定目录查找到生成的日志文件;
  • 本地idea调试时,推荐Grep Console插件可实现控制台的自定义颜色输出
  • 原文链接:https://mp.weixin.qq.com/s/kPb5NYizLf-svFaBrRkcuQ

    郑重声明:本文内容及图片均整理自互联网,不代表本站立场,版权归原作者所有,如有侵权请联系管理员(admin#wlmqw.com)删除。
    (0)
    用户投稿
    上一篇 2022年6月15日
    下一篇 2022年6月15日

    相关推荐

    • 20年老IP首次摆脱“渣画质”?可用户买账吗?

      300亿市场规模的传奇类游戏,在产品层面普遍“躺平”。 但最近,却出现了一个不惜重金升级美术表现的“搅局者”。 一款由UE4引擎打造的《传奇天下》近期上线,首日表现尚可(免费榜第4…

      2022年8月1日
    • 挑战全网最烂的鼠标

      我旅行的时间很长,旅途也是很长的——泰戈尔 直击心灵深处 身份信息显示,这是赛睿sensei RAW 因为最近吃土了,所以需要挖土,所以得到了这货,经典的赛睿SENSEI RAW鼠…

      2022年8月21日
    • Model3降到21万?特斯拉的法务部怎么还不出手!

      不好意思,这次我支持特斯拉的法务部出来扔律师函。 说Model3准备降到21.99万的,有一个算一个都在散播谣言,扰乱消费市场秩序。日前,有消息表示特斯拉Model3价格将会从现在…

      2022年6月21日
    • 美调查:美国超半数老年人生活贫困 通胀加剧财务压力

      海外网7月26日电据美国《财富》杂志网站报道,美国马萨诸塞大学波士顿分校日前发布一份针对老年人的调查,数据显示,美国超半数老年人生活在贫困之中。高通胀提升了老年人的生活成本,更加剧…

      2022年8月4日
    • 四川:涉水景区及滩涂峡谷等要落实专人盯守巡查

      四川省防办: 涉水景区及滩涂峡谷等要落实专人盯守巡查 记者8月14日从四川省应急管理厅获悉,四川省防办印发了《关于深刻汲取教训扎实做好局地短时强降雨防范应对工作的紧急通知》(以下简…

      2022年8月16日
    • 学会7个方法,让孩子爱上蔬菜

      湖南日报全媒体记者 王铭俊 通讯员 袁欢 蔬菜是日常饮食中不可缺少的食物之一,但不少孩子只吃肉不吃菜或者只吃水果不吃菜。 为什么孩子不爱吃蔬菜,有哪些办法能让孩子爱上蔬菜?今天,湖…

      2022年8月19日
    • 伟人女儿指出教材错误,老师却让她叫家长,最终毛主席如何处置?

      这可能是世界上面子最大的班主任,不仅当场冤枉李讷,甚至还要求李讷的父亲来办公室谈话。殊不知,李讷的父亲,正是伟大领袖毛主席。 受到批评的李讷无法反驳老师,又不愿意透露父亲的身份,只…

      2022年8月25日
    • 殷桃真人到底长啥样?当看到路人镜头下的她后,网友:确定有41?

      殷桃算是一位颜值和气质都很好的女演员了,她一直以来在演艺圈中的表现都很不错。她的很多经典作品也让观众难以忘怀,只是如今的殷桃热度已经大不如前,即便是有一些比较出色的角色,也很快就被…

      2022年5月15日
    • 光场:计算光学的灵魂

      前 言 计算光学的本质是光场的获取与解译,无疑,光场扮演着非常重要的角色。随着研究的深入,我们发现:光场是作为计算成像的灵魂存在的。 光场的本质是光的物理属性在空间和时间维度上的分…

      2022年8月15日
    • 法士特入选国家级智能制造标准应用试点项目名单

      日前,工业和信息化部公示了2022年度智能制造标准应用试点项目名单,法士特申报的“商用车传动系统智能工厂标准应用试点”成功入选。 法士特依托先进制造技术与新一代信息技术的深度融合,…

      2022年8月11日

    联系我们

    联系邮箱:admin#wlmqw.com
    工作时间:周一至周五,10:30-18:30,节假日休息