Spring Boot 全局异常拦截,并记录日志

实现代码

主要功能实现类

import com.alibaba.fastjson.JSON;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.web.bind.annotation.ControllerAdvice;
import org.springframework.web.bind.annotation.ExceptionHandler;

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.FileWriter;
import java.io.IOException;
import java.io.OutputStream;
import java.io.PrintWriter;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.Enumeration;

/**
 * Created by tt on 2017/6/16.
 * 全局异常处理类
 */
@ControllerAdvice
public class ExceptionHandlerAdvice {

    private Logger logger = LoggerFactory.getLogger(getClass());

    @Value("${err.logging.file}")
    private String errFile;

    private final SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");

    /**
     * 拦截所有的Exception
     */
    @ExceptionHandler(value = {Exception.class})
    public void exception(Exception exception, HttpServletResponse response, HttpServletRequest request) {
        try {
            FileWriter fileWriter = new FileWriter(errFile, true);
            fileWriter.write("========================================================================================================================================\n");
            //记录错误时间
            fileWriter.write(sdf.format(new Date()) + "\n");
            //记录用户设备型号
            fileWriter.write(request.getHeader("User-Agent") + "\n");

            String ip = getRemoteAddress(request);
            String requestURL = request.getRequestURL().toString();
            String url = request.getQueryString() == null ? requestURL + "" : (requestURL + "?" + request.getQueryString());
            //记录请求地址
            fileWriter.write(url + "\n");
            //记录请求ip地址
            fileWriter.write(ip + "\n");
            Enumeration paramNames = request.getParameterNames();
            while (paramNames.hasMoreElements()) {
                String paramName = (String) paramNames.nextElement();
                String[] paramValues = request.getParameterValues(paramName);
                if (paramValues.length == 1) {
                    String paramValue = paramValues[0];
                    if (paramValue.length() != 0) {
                        //记录请求参数
                        fileWriter.write("param: " + paramName + " = " + paramValue + "\n");
                    }
                }
            }
            PrintWriter printWriter = new PrintWriter(fileWriter);
            //把异常信息记录到日志文件中
            exception.printStackTrace(printWriter);
            //关闭writer流
            fileWriter.write("\n");
            fileWriter.close();
            printWriter.close();
        } catch (IOException e) {
            logger.info("I can't open the file " + errFile);
        }
        //控制台打印异常详细信息
        exception.printStackTrace();
        //使用OutputStream流向客户端输出错误信息
        try {
            OutputStream os = response.getOutputStream();
            //通过设置响应头控制浏览器以UTF-8的编码显示数据,如果不加这句话,那么浏览器显示的将是乱码
            response.setHeader("content-type", "text/html;charset=UTF-8");
            ResultData<Object> resultData = new ResultData<>();
            String e = exception.toString();
            // 异常类型
            resultData.setData("异常类型:" + (e.contains(":") ? e.substring(0, e.indexOf(":")) : e));
            resultData.setCode(555);// 异常信息
            resultData.setMsg("异常信息:" + exception.getMessage());
            String data = JSON.toJSONString(resultData);
            //将字符转换成字节数组,指定以UTF-8编码进行转换
            byte[] dataByteArr = data.getBytes("UTF-8");
            //使用OutputStream流向客户端输出字节数组
            os.write(dataByteArr);
            //关闭输出流
            os.close();
        } catch (IOException e1) {
            e1.printStackTrace();
        }
    }

    private static String getRemoteAddress(HttpServletRequest request) {
        String ip = request.getHeader("x-forwarded-for");
        if (ip == null || ip.length() == 0 || ip.equalsIgnoreCase("unknown")) {
            ip = request.getHeader("Proxy-Client-IP");
        }
        if (ip == null || ip.length() == 0 || ip.equalsIgnoreCase("unknown")) {
            ip = request.getHeader("WL-Proxy-Client-IP");
        }
        if (ip == null || ip.length() == 0 || ip.equalsIgnoreCase("unknown")) {
            ip = request.getRemoteAddr();
        }
        return ip;
    }
}

以下为ResultData类,用于返回给客户端的json格式数据

public class ResultData<T> {

    private T data;

    private Integer code = 200;

    private String msg;

    private Boolean success = true;

    // 此处省略getter  setter
}

代码说明

@ControllerAdvice

通过@ControllerAdvice注解到类上,我们可以将对于控制器的全局配置放在同一个位置,通过@ExceptionHandler,@InitBinder,@ModelAttribute(本文只提到@ExceptionHandler,另外两个注解读者有兴趣可自行尝试)注解在该类的方法上,这对所有注解了@Controller的类内的注解了@RequestMapping的方法都有效。

我们现在要用到的是@ExceptionHandler,用于全局处理控制器里的异常。

@ExceptionHandler(value = {Exception.class})

这边可以看到@ExceptionHandler注解内我们使用到了value属性,它可以过滤拦截的条件。此处我们拦截所有的Exception。除此之外我们还可以根据自己的需要,定制细化的异常处理,把各种不同的异常类型分开处理。

@Value

@Value("${err.logging.file}")

@Value 用于获取 application.properties 文件的外部配置属性,此示例在外部配置文件内的配置如下,主要用于配置异常处理的日志错误信息保存的路径,方便项目运行动态设置。

err.logging.file = err.log

其他细节

OutputStream os = response.getOutputStream();
//通过设置响应头控制浏览器以UTF-8的编码显示数据,如果不加这句话,那么浏览器显示的将是乱码
response.setHeader("content-type", "text/html;charset=UTF-8");
ResultData<Object> resultData = new ResultData<>();
String e = exception.toString();
// 异常类型
resultData.setData("异常类型:" + (e.contains(":") ? e.substring(0, e.indexOf(":")) : e));
resultData.setCode(555);// 异常信息
resultData.setMsg("异常信息:" + exception.getMessage());
String data = JSON.toJSONString(resultData);
//将字符转换成字节数组,指定以UTF-8编码进行转换
byte[] dataByteArr = data.getBytes("UTF-8");
//使用OutputStream流向客户端输出字节数组
os.write(dataByteArr);
//关闭输出流
os.close();

这里用于异常捕获处理日志后,向客户端输出的异常信息,处理后的json输出示例:

{
  code: 555,
  data: "异常类型:org.springframework.web.bind.MissingServletRequestParameterException",
  msg: "异常信息:Required List parameter 'ids' is not present",
  success: false
}

这样可以保持客户端请求服务端接口的返回数据格式一致。

最后附上日志记录信息的结果(具体的记录信息科自行定制,本文仅提供方案)

屏幕快照 2017-08-16 13.33.29.png
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 200,176评论 5 469
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 84,190评论 2 377
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 147,232评论 0 332
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 53,953评论 1 272
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 62,879评论 5 360
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,177评论 1 277
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,626评论 3 390
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,295评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,436评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,365评论 2 317
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,414评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,096评论 3 315
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,685评论 3 303
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,771评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,987评论 1 255
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,438评论 2 346
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,032评论 2 341

推荐阅读更多精彩内容