Java自学者论坛

 找回密码
 立即注册

手机号码,快捷登录

恭喜Java自学者论坛(https://www.javazxz.com)已经为数万Java学习者服务超过8年了!积累会员资料超过10000G+
成为本站VIP会员,下载本站10000G+会员资源,会员资料板块,购买链接:点击进入购买VIP会员

JAVA高级面试进阶训练营视频教程

Java架构师系统进阶VIP课程

分布式高可用全栈开发微服务教程Go语言视频零基础入门到精通Java架构师3期(课件+源码)
Java开发全终端实战租房项目视频教程SpringBoot2.X入门到高级使用教程大数据培训第六期全套视频教程深度学习(CNN RNN GAN)算法原理Java亿级流量电商系统视频教程
互联网架构师视频教程年薪50万Spark2.0从入门到精通年薪50万!人工智能学习路线教程年薪50万大数据入门到精通学习路线年薪50万机器学习入门到精通教程
仿小米商城类app和小程序视频教程深度学习数据分析基础到实战最新黑马javaEE2.1就业课程从 0到JVM实战高手教程MySQL入门到精通教程
查看: 11428|回复: 0

从零开始学 Java - Spring MVC 统一异常处理

[复制链接]
  • TA的每日心情
    奋斗
    2025-3-18 14:43
  • 签到天数: 805 天

    [LV.10]以坛为家III

    2053

    主题

    2111

    帖子

    73万

    积分

    管理员

    Rank: 9Rank: 9Rank: 9

    积分
    731050
    发表于 2021-4-22 09:35:34 | 显示全部楼层 |阅读模式

    看到 Exception 这个单词都心慌

    如果有一天你发现好久没有看到Exception这个单词了,那你会不会想念她?我是不会的。她如女孩一样的令人心动又心慌,又或者你已经练功到了孤独求败,等了半辈子终于看到了她,这时候你的小弟准备冲上去解决它说:大哥,我来解决它。你摆摆手说:还是我来最后一次吧...

    从此,你再也没看到过Exception了。

    异常是个好东西

    做开发的对异常在熟悉不过了,几乎是天天打交道的。记得当年写代码的时候记住的第一个异常是未将对象引用设置到对象的实例,现在想想已经好久没有看到他了,当然这是 .Net 下的一个异常,对应 Java 下的是java.lang.NullPointerException

    其实,异常并不可怕,可怕的是你不知道怎么解决它。 解决异常的方式有很多种,比如写好每一行代码,保证不出任何逻辑错误,就可以从根本上解决问题,但是,没有一个程序员能保证自己的代码不报异常,这也就是为什么会有 Bug 这个令人讨厌的东西了。当然,如果我们写的代码都没有异常,不出 Bug ,那怎么让我们有和测试妹子在一起工作的机会呢!哈哈哈...

    既然我们保证不了从根本上解决代码不出异常的情况,那我们是谁阿,我们为了在测试妹子面前展示厉害到无敌的敲代码能力,我们就偷偷的在业务代码里使用 try...catch... 来让她看不出你写代码能力不行,你这时候就厉害到不行了,你脉脉看着她,她看着你...

    难道不应该有即使我们不使用 try...catch... 也能捕获写出来的异常么?

    这里有一本《异常秘籍》

    并不是说使用 try...catch... 不行,是因为在程序里可能会有你可预知的异常,这时候你当然会去使用它来捕获异常,但是如果在一个你不可预知的方法里,你本能认为它不会出异常,你还会使用 try...catch... 去捕获么?如果你回答会,那我问你:你意思就是在每一个方法里加入 try...catch... 代码了?

    当然,这样并不是不行,是不太好。你想想,这样写起来是不是也太累了,我们都很懒的,而且我们程序里有一个讲究是「耦合性」,那你这就完全不符合「高内聚、低耦合」咯?简单说,异常的处理对业务代码的侵入性太强了,不够美,我们当然有更好的处理方式了。

    我们可不可以统一处理异常呢? 当然可以啦!

    现在就新建一个ExceptionHandler.java异常处理类。

    @Component
    public class ExceptionHandler implements HandlerExceptionResolver {
    
        private static Logger log = Logger.getLogger(ExceptionHandler.class);
    
        @Override
        public ModelAndView resolveException(HttpServletRequest request, HttpServletResponse response, Object handler, Exception exception) {
            log.error("ExceptionHandler 捕获的异常:", exception);
    
            String requestType = request.getHeader("X-Requested-With");
            String type = "api";    //TODO:
            if (!type.equals("api") && StrUtil.isNullOrEmpty(requestType)) {
                // 非API请求
                return new ModelAndView("redirect:/500.html");
            } else {// JSON格式返回
    
                Map<String, Object> responseMap = new HashMap<String, Object>();
                responseMap.put("code", -1);
                responseMap.put("msg", "系统异常,请稍后重试!");
                String json = new Gson().toJson(responseMap);
                response.setCharacterEncoding("UTF-8");
                response.setContentType("application/json; charset=utf-8");
                try {
                    response.getWriter().write(json);
                    response.getWriter().flush();
                    return null;
                } catch (IOException e) {
                    log.error("", e);
                }
            }
            return new ModelAndView("redirect:/500.html");
        }
    }
    

    好,就这样简单,完成了异常的统一处理。其实是 Spring 中定义的 HandlerExceptionResolver 接口,我们重写里边的 resolveException 方法就可以捕获项目中未处理的异常。当然,这就要求我们项目中的异常要一层一层的抛出去了,这时候不要担心,我们会最终捕获它的。

    现在简单说说上边我的捕获代码的处理方式,首先,我捕获到异常就写一个 log 记录它,以便于我们找出查看,然后,如果是我们的 API 接口请求的话,我就返回接口的统一 Json 格式,如果是其他请求的话,我就会返回到一个 500 的错误页面,以优雅的方式提示用户。

    怎么使用

    其实,完全就不用说怎么使用了,非常简单的用法,mafly.那这里我就试着抛一个异常出去,然后故意不捕获它,看看结果到底会怎样?
    1.先在 ServiceImp 层抛一个异常。
    exception_serviceimp.jpg

    2.在 Controller 层调用这个方法。
    exception_controller.jpg

    3.请求一个 API 接口。
    exception_log.jpg

    这时候,你看控制台打印出来了异常日志,就是我们刚刚抛出来的,你也可以调试一下,看看执行过程。这个时候,你就可以随心所欲的统一处理异常了。
    文章的具体的案例,都可以访问我的 Github 看到 https://github.com/mafly/SpringDemo

    哎...今天够累的,签到来了1...
    回复

    使用道具 举报

    您需要登录后才可以回帖 登录 | 立即注册

    本版积分规则

    QQ|手机版|小黑屋|Java自学者论坛 ( 声明:本站文章及资料整理自互联网,用于Java自学者交流学习使用,对资料版权不负任何法律责任,若有侵权请及时联系客服屏蔽删除 )

    GMT+8, 2025-4-21 19:41 , Processed in 0.087459 second(s), 29 queries .

    Powered by Discuz! X3.4

    Copyright © 2001-2021, Tencent Cloud.

    快速回复 返回顶部 返回列表