8.统一异常处理 + 统一记录日志

目录

1.统一异常处理

2.统一记录日志


1.统一异常处理


在 HomeController 类中添加请求方法(服务器发生异常之后需要统一处理异常,记录日志,然后转到 500 页面,需要人工处理重定向到 500 页面,提前把 500 页面请求访问配置)

    @RequestMapping(path = "/error", method = RequestMethod.GET)public String getErrorPage() {return "/error/500";}

在 controller 类下新建 advice 包,创建 ExceptionAdvice 类:

  • 添加注解 @ControllerAdvice 统一处理异常:此时这个组件会扫面所有的 Bean,做一个限制 @ControllerAdvice(annotations = Controller.class):这个组件只去扫面带有 Controller 注解的 Bean
  • 添加所有方法处理所有错误情况:添加注解 @ExceptionHandler 表示这个方法处理所有异常的方法—— @ExceptionHandler({Exception.class})
  • 通常参数为 Exception e, HttpServletRequest request, HttpServletResponse response
  • 注入日志,并且把异常记录日志,想要把异常非常详细的栈的信息记录:遍历栈的信息得到的是数组(每次遍历得到一条异常信息,打印日志)
  • 然后给浏览器响应(重定向到错误页面)
  • 这时候还需要判断这个请求是普通请求还是异步请求:浏览器访问服务器可能是普通请求(希望返回网页,然后重定向到 500)也可能是异步请求(希望返回 JSON,不可以返回到页面 HTML)
  • 通过 request.getHeader("x-requested-with") 获取请求返回 String:如果返回值等于 "XMLHttpRequest"则表示为异步请求,这个请求是以 XML 的形式访问,希望返回 XML,只有异步请求才希望返回 XML(普通请求返回 HTML)然后响应字符串("application/plain;charset=utf-8":向浏览器返回普通字符串,可以是 JSON 格式,需要人为的将字符串转化为 JS 对象),获取输出流输出JSON字符串;如果是普通请求,重定向到错误页面(获取项目访问路径 + "/error" 路径)
package com.example.demo.controller.advice;import com.example.demo.util.CommunityUtil;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.stereotype.Controller;
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.IOException;
import java.io.PrintWriter;//添加注解 @ControllerAdvice 统一处理异常:此时这个组件会扫面所有的 Bean,
//做一个限制 @ControllerAdvice(annotations = Controller.class):这个组件只去扫面带有 Controller 注解的 Bean
@ControllerAdvice(annotations = Controller.class)
public class ExceptionAdvice {private static final Logger logger = LoggerFactory.getLogger(ExceptionAdvice.class);//添加所有方法处理所有错误情况:添加注解 @ExceptionHandler 表示这个方法处理所有异常的方法@ExceptionHandler({Exception.class})public void handleException(Exception e, HttpServletRequest request,HttpServletResponse response) throws IOException {//注入日志,并且把异常记录日志,想要把异常非常详细的栈的信息记录// 遍历栈的信息得到的是数组(每次遍历得到一条异常信息,打印日志)logger.error("服务器发生异常: " + e.getMessage());for (StackTraceElement element : e.getStackTrace()) {logger.error(element.toString());}//然后给浏览器响应(重定向到错误页面)//这时候还需要判断这个请求是普通请求还是异步请求:浏览器访问服务器可能是普通请求(希望返回网页,然后重定向到 500)// 也可能是异步请求(希望返回 JSON,不可以返回到页面 HTML)//通过 request.getHeader("x-requested-with") 获取请求返回 String:如果返回值等于 "XMLHttpRequest"则表示为异步请求,// 这个请求是以 XML 的形式访问,希望返回 XML,只有异步请求才希望返回 XML(普通请求返回 HTML)String xRequestedWith = request.getHeader("x-requested-with");if ("XMLHttpRequest".equals(xRequestedWith)) {// 然后响应字符串("application/plain;charset=utf-8":向浏览器返回普通字符串//可以是 JSON 格式,需要人为的将字符串转化为 JS 对象)response.setContentType("application/plain;charset=utf-8");//获取输出流输出JSON字符串;如果是普通请求,重定向到错误页面(获取项目访问路径 + "/error" 路径)PrintWriter writer = response.getWriter();writer.write(CommunityUtil.getJSONString(1, "服务器异常!"));} else {response.sendRedirect(request.getContextPath() + "/error");}}
}

2.统一记录日志

  • 可不可以利用控制器通知统一处理?控制器通知在发生异常时可以统一处理,而统一记录日志并不是发生异常才记录
  • 可不可以使用拦截器?拦截器也是针对控制器做处理,而记录日志并不是只有在控制器中记录(目前我们对业务组件做日志记录)
  • 使用传统方法:把记录日志的内容封装到组件中,在不同的业务组件(service)方法中调用(比如在一开始记录日志,在写也业务方法之前记录日志)。这种方法也有弊端:业务组件方法是处理业务,但是在之前添加日志需求(不是业务需求,是系统需求),所以在业务方法中耦合了系统需求是有弊端的,例如现在系统需求发生变化,不想在业务之前记录日志,而是在之后记录日志,改动比较大,很麻烦
  • 解决问题的最好方法:将记录日志(系统需求)单独实现——使用 AOP,单独定义一个组件,不和业务组件发生直接关系,将业务组件通用逻辑封装在这个组件中

对于 AOP 的知识可以阅读之前学习的 AOP 博客:Spring AOP-CSDN博客

在 pom.xml 添加依赖:

<!-- https://mvnrepository.com/artifact/org.springframework.boot/spring-boot-starter-aop -->
<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-aop</artifactId>
</dependency>

对所有业务组件记录日志(在业务组件一开始记录日志),新建 aspect 包创建 ServiceLogAspect 类(存放切面主键):

  • 添加注解 @Component、@Aspect
  • 实例化 Logger
  • 声明切点:所有业务组件都去处理
  • 使用前置通知在业务组件一开始记录日志
  • 记录格式:用户[1.2.3.4],在[xxx],访问了[com.example.demo.service.xxx()].
  • 用户 ip 通过 request 获取,获取 request:RequestContextHolder.getRequestAttributes();
  • 拼接时间:new Date,然后实例化
  • 访问某个类某个方法(类名 + 方法名):给方法添加 JoinPoint 连接点参数,连接点指代程序植入的目标方法
  • 最后再进行全部拼接
package com.example.demo.aspect;import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.stereotype.Component;
import org.springframework.web.context.request.RequestContextHolder;
import org.springframework.web.context.request.ServletRequestAttributes;import javax.servlet.http.HttpServletRequest;
import java.text.SimpleDateFormat;
import java.util.Date;@Component
@Aspect
public class ServiceLogAspect {//实例化 Loggerprivate static final Logger logger = LoggerFactory.getLogger(ServiceLogAspect.class);//声明切点:所有业务组件都去处理@Pointcut("execution(* com.example.demo.service.*.*(..))")public void pointcut() {}//使用前置通知在业务组件一开始记录日志@Before("pointcut()")public void before(JoinPoint joinPoint) {//记录格式:用户[1.2.3.4],在[xxx],访问了[com.example.demo.service.xxx()].//用户 ip 通过 request 获取,获取 request:RequestContextHolder.getRequestAttributes();ServletRequestAttributes attributes =(ServletRequestAttributes) RequestContextHolder.getRequestAttributes();HttpServletRequest request = attributes.getRequest();String ip = request.getRemoteHost();//拼接时间:new Date,然后实例化String now = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss").format(new Date());//访问某个类某个方法(类名 + 方法名):给方法添加 JoinPoint 连接点参数,连接点指代程序植入的目标方法String target = joinPoint.getSignature().getDeclaringTypeName() + "." + joinPoint.getSignature().getName();//全部拼接logger.info(String.format("用户[%s],在[%s],访问了[%s].", ip, now, target));}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/222905.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

顶级Mac数据恢复工具—— 13个 Mac 数据恢复程序榜单

如果您点击此博客&#xff0c;首先可能是您不小心格式化了外部或内部存储&#xff0c;无论是 SD卡还是硬盘&#xff0c;其次&#xff0c;您收到了一些错误消息&#xff0c;表明您丢失了所有内容&#xff0c;现在您已经精疲力竭了的形状。原因可能有多种&#xff1a;您不小心删除…

linux 搭建Nginx网页(编译安装)

♥️作者&#xff1a;小刘在C站 ♥️个人主页&#xff1a; 小刘主页 ♥️不能因为人生的道路坎坷,就使自己的身躯变得弯曲;不能因为生活的历程漫长,就使求索的 脚步迟缓。 ♥️学习两年总结出的运维经验&#xff0c;以及思科模拟器全套网络实验教程。专栏&#xff1a;云计算技…

探索 Linux vim/vi 编辑器:介绍、模式以及基本操作演示

&#x1f490;作者&#xff1a;insist-- &#x1f490;个人主页&#xff1a;insist-- 的个人主页 理想主义的花&#xff0c;最终会盛开在浪漫主义的土壤里&#xff0c;我们的热情永远不会熄灭&#xff0c;在现实平凡中&#xff0c;我们终将上岸&#xff0c;阳光万里 ❤️欢迎点…

C++中类的静态成员、存储、this、友元和运算符重载

静态成员 在类定义中&#xff0c;它的成员&#xff08;包括成员变量和成员函数&#xff09;&#xff0c;这些成员可以用关键字static 声明为静态的&#xff0c;称为静态成员。 不管这个类创建了多少个对象&#xff0c;静态成员只有一个拷贝&#xff0c;这个拷贝被所有属于这个…

Qt 软件开发框架(主要部分)

目录 1、 一个软件基本要素 &#xff08;1&#xff09;UI模块 &#xff08;2&#xff09;网络模块 &#xff08;3&#xff09;业务逻辑模块 &#xff08;4&#xff09;中间层 &#xff08;5&#xff09;独立模块&#xff08;守护进程、更新模块、日志收集模块…&#xff…

1233:单词倒置(C语言)

题目描述 最近birdfly收到了女友的几份信件&#xff0c;为了只要他俩知道信件的秘密&#xff0c;女友把信件里的每个单词都倒置了。这样只有birdfly将它们倒置过来才能明白女友的心思了。为此birdfly还特意请你编写程序帮他解决一下这个问题。 简单起见假定每封信只包含英文单词…

【AD9510 概要总结】A..

目录 特征FEATURES概述 GENERAL DESCRIPTION功能描述 FUNCTIONAL DESCRIPTIONPLL部分REFIN PLL参考输入—REFINVCO/VCXO时钟输入—CLK2PLL基准分频器—RVCO/VCXO 反馈分频器—N (P, A, B)A 和 B 计数器确定 P、A、B 和 R 的值 鉴频鉴相器&#xff08;PFD&#xff09;和电荷泵消…

JavaScript的学习

HTML的学习-CSDN博客 从html的学习中 其实我已经用到了 JavaScript的脚本 &#xff08;GPT&#xff09; 例如 echo <script>alert("账号密码错误"); window.location"index.html";</script>; 弹窗 然后定位到 index.html 这里能够让我们更…

eutil.dll文件缺失修复全指南,教你快速修复eutil.dll

eutil.dll缺失了要怎么办&#xff1f;eutil.dll是一种常见的动态链接库&#xff08;DynamicLinkLibrary&#xff0c;DLL&#xff09;文件&#xff0c;它在Windows操作系统中发挥着重要作用。DLL文件允许程序共享代码以执行诸如打印或连接网络之类的功能。这不仅节省了系统资源&…

使用群晖Synology Office提升生产力:如何多人同时编辑一个文件

使用群晖Synology Office提升生产力&#xff1a;多人同时编辑一个文件 正文开始前给大家推荐个网站&#xff0c;前些天发现了一个巨牛的人工智能学习网站&#xff0c;通俗易懂&#xff0c;风趣幽默&#xff0c;忍不住分享一下给大家。点击跳转到网站。 文章目录 使用群晖Synol…

时间序列预测 — Informer实现多变量负荷预测(PyTorch)

目录 1 实验数据集 2 如何运行自己的数据集 3 报错分析 1 实验数据集 实验数据集采用数据集4&#xff1a;2016年电工数学建模竞赛负荷预测数据集&#xff08;下载链接&#xff09;&#xff0c;数据集包含日期、最高温度℃ 、最低温度℃、平均温度℃ 、相对湿度(平均) 、降雨…

智能优化算法应用:基于蜉蝣算法无线传感器网络(WSN)覆盖优化 - 附代码

智能优化算法应用&#xff1a;基于蜉蝣算法无线传感器网络(WSN)覆盖优化 - 附代码 文章目录 智能优化算法应用&#xff1a;基于蜉蝣算法无线传感器网络(WSN)覆盖优化 - 附代码1.无线传感网络节点模型2.覆盖数学模型及分析3.蜉蝣算法4.实验参数设定5.算法结果6.参考文献7.MATLAB…