使用Springboot AOP进行请求接口异常监控
常用注解
@Aspect 切面类
@Before 前置
@AfterReturning 后置
@Around 环绕
@AfterThrowing 异常
切入点设置
execution(public * *(..)) 定义任意公共方法的执行
execution(* set*(..)) 定义任何一个以"set"开始的方法的执行
execution(* com.sys.service.UserService.*(..)) 定义UserService 接口的任意方法的执行
execution(* com.sys.service.*.*(..)) 定义在service包里的任意方法的执行
execution(* com.sys.service ..*.*(..)) 定义在service包和所有子包里的任意类的任意方法的执行
execution(* com.sys.pointcutexp…JoinPointObject.*(…)) 定义在pointcutexp包和所有子包里的JoinPointObject类的任意方法的执行
本次使用aop用途说明
使用@AfterReturning方法拿到请求接口的返回状态码,将需要处理的问题状态码及错误信息保存到数据库及输出到日志文件,方便通过数据库进行接口调用错误信息监控。通过配置切入点对controller的所有请求接口进行处理,就无需再在每一个请求方法中进行处理。
使用@AfterThrowing方法拿到请求接口方法异常调用信息,其他同上。
参考代码
package com.system.common.aop;import com.alibaba.fastjson.JSON;
import com.system.service.SysLogService;
import com.xlx.entity.BaseResponse;
import com.xlx.entity.bsc.system.DO.SystemLogInfoDO;
import com.xlx.utils.ExceptionUtil;
import com.xlx.utils.StringParseUtils;
import lombok.extern.slf4j.Slf4j;
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.*;
import org.aspectj.lang.reflect.MethodSignature;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;import java.lang.reflect.Method;
import java.util.Arrays;/*** @version: java version 1.8* @Author: sagitario* @description: 异常请求记录到数据库及日志输出* @date: 2023-04-20 11:44*/
@Component
@Aspect
@Slf4j
public class ApiResponseAOP {@Autowiredprivate SysLogService sysLogService;private static String [] passUrl = {"过滤的请求url"};@Pointcut("execution(* com.*.controller.*.*(..))")private void methodAspect(){}//定义一个切入点@Before("methodAspect()")public void doAudit(JoinPoint joinPoint) {log.info("BeforeAdvice.............");}@AfterReturning(value = "methodAspect()",returning = "methodResult")public void afterReturning(JoinPoint joinPoint, Object methodResult) {MethodSignature ms = (MethodSignature) joinPoint.getSignature();Method method = ms.getMethod();String clazz = method.getDeclaringClass().getName();log.debug("请求类为:" + clazz);String mName = method.getName();log.debug("请求方法为:" + mName);if(Arrays.binarySearch(passUrl, mName)<0){BaseResponse baseResponse = (BaseResponse) methodResult;if(!baseResponse.isSuccess()){StringBuffer stringBuffer = new StringBuffer();for (int i = 0; i < joinPoint.getArgs().length; i++) {log.debug("请求参数为:" + JSON.toJSONString(joinPoint.getArgs()[i]));stringBuffer.append(JSON.toJSONString(joinPoint.getArgs()[i]));}log.debug("请求返回内容为:" + methodResult.toString());SystemLogInfoDO systemLogInfoDO = new SystemLogInfoDO();systemLogInfoDO.setInfo(clazz + "." + mName);systemLogInfoDO.setParam(stringBuffer.toString());systemLogInfoDO.setCode(baseResponse.getCode());systemLogInfoDO.setMessage(baseResponse.getMessage());if(baseResponse.getCode()==204){systemLogInfoDO.setLevel("INFO");}else{systemLogInfoDO.setLevel("ERROR");}sysLogService.insertSelective(systemLogInfoDO);}}}@AfterThrowing(value = "methodAspect()",throwing = "e")public void afterThrowing(JoinPoint joinPoint, Exception e) {log.info("AfterThrowing advice");MethodSignature ms = (MethodSignature) joinPoint.getSignature();Method method = ms.getMethod();String clazz = method.getDeclaringClass().getName();log.info("请求类为:" + clazz);String mName = method.getName();log.info("请求方法为:" + mName);SystemLogInfoDO systemLogInfoDO = new SystemLogInfoDO();try {StringBuffer stringBuffer = new StringBuffer();for (int i = 0; i < joinPoint.getArgs().length; i++) {log.info("请求参数为:" + JSON.toJSONString(joinPoint.getArgs()[i]));stringBuffer.append(JSON.toJSONString(joinPoint.getArgs()[i]));}systemLogInfoDO.setParam(stringBuffer.toString());}catch (Exception e1){log.info("请求方法参数无法解析:" + ExceptionUtil.getMessage(e1));systemLogInfoDO.setParam("请求方法参数无法解析");}String error = ExceptionUtil.getMessage(e);log.info("afterThrowing异常:{}",error);systemLogInfoDO.setInfo(clazz + "." + mName);systemLogInfoDO.setCode(500);systemLogInfoDO.setMessage(error);systemLogInfoDO.setLevel("EXCEPTION");sysLogService.insertSelective(systemLogInfoDO);}}
日志输出
2023-05-22 18:36:14 [http-nio-8831-exec-6] [com.loan.common.aop.ApiResponseAOP:82] INFO com.loan.common.aop.ApiResponseAOP - 请求类为:com.loan.controller.RepayController
2023-05-22 18:36:14 [http-nio-8831-exec-6] [com.loan.common.aop.ApiResponseAOP:84] INFO com.loan.common.aop.ApiResponseAOP - 请求方法为:repaymentRecord
2023-05-22 18:36:14 [http-nio-8831-exec-6] [com.loan.common.aop.ApiResponseAOP:89] INFO com.loan.common.aop.ApiResponseAOP - 请求参数为:"110014"
2023-05-22 18:36:14 [http-nio-8831-exec-6] [com.xlx.exception.GlobalExceptionHandler:26] ERROR com.xlx.exception.GlobalExceptionHandler - feign.RetryableException: Read timed out executing GET http://external-platform-server/repay/repaymentRecordsat feign.FeignException.errorExecuting(FeignException.java:213)at feign.SynchronousMethodHandler.executeAndDecode(SynchronousMethodHandler.java:115)......
2023-05-22 18:36:14 [http-nio-8831-exec-6] [com.loan.common.aop.ApiResponseAOP:98] INFO com.loan.common.aop.ApiResponseAOP - afterThrowing异常:feign.RetryableException: Read timed out executing GET http://external-platform-server/repay/repaymentRecordsat feign.FeignException.errorExecuting(FeignException.java:213)at feign.SynchronousMethodHandler.executeAndDecode(SynchronousMethodHandler.java:115)at feign.SynchronousMethodHandler.invoke(SynchronousMethodHandler.java:80)at feign.ReflectiveFeign$FeignInvocationHandler.invoke(ReflectiveFeign.java:103)
其他
JDK动态代理相关可参考 一篇文章彻底搞懂java AOP、@Before、@After、@AfterReturning、@AfterThrowing、@Around的使用、Spring AOP详解
相关文章:
使用Springboot AOP进行请求接口异常监控
常用注解 Aspect 切面类 Before 前置 AfterReturning 后置 Around 环绕 AfterThrowing 异常 切入点设置 execution(public * *(..)) 定义任意公共方法的执行 execution(* set*(..)) 定义任何一个以"set"开始的方法的执行 execution(* com.sys.service.UserService…...
【云原生|Kubernetes】05-Pod的存储卷(Volume)
【云原生Kubernetes】05-Pod的存储卷(Volume) 文章目录 【云原生Kubernetes】05-Pod的存储卷(Volume)简介Volume类型解析emptyDirHostPathgcePersistentDiskNFSiscsiglusterfsceph其他volume 简介 Volume 是Pod 中能够被多个容器访问的共享目录。 Kubern…...
Python实现数据结构
文章目录 一、Python实现数据结构1.1 python实现单向链表1.2 python实现单向循环链表1.3 python实现双向链表 一、Python实现数据结构 1.1 python实现单向链表 singleLinkedList.py class SingleNode:"""the node of single link list"""def …...
esp32CAM环境安装教程---串口驱动安装
前言 (1)本人安装好arduino 的ESP32环境之后, 发现一直下载不进去程序。一直说Cannot configure port, something went wrong. Original message: PermissionError。 (2)查阅了很多资料,用了各种办法&#…...
Java中List和Array转换
文章目录 List -> Array1. 调用toArray()方法直接返回一个Object[]数组:2. 给toArray(T[])传入一个类型相同的Array,List内部自动把元素复制到传入的Array中:3. 通过List接口定义的T[] toArray(IntFunction<T[]> generator)方法&…...
如何能确定数据库中root用户的密码是什么
如果您无法确定数据库中 root 用户的密码,有几种方法可以尝试找回或重置密码: 1. 使用已知密码:如果您有之前设置的 root 用户密码,可以使用该密码进行登录。 2. 查找密码文件:在某些情况下,MariaDB 可能…...
由浅入深Netty协议设计与解析
目录 1 为什么需要协议?2 redis 协议举例3 http 协议举例4 自定义协议要素4.1 编解码器4.2 什么时候可以加 Sharable 1 为什么需要协议? TCP/IP 中消息传输基于流的方式,没有边界。 协议的目的就是划定消息的边界,制定通信双方要…...
iptables防火墙(1)
iptables防火墙 一、iptables概述二、netfilter/iptables 关系三、四表五链1.四表2.五链 四、规则链之间的匹配顺序五、规则链内的匹配顺序六、iptables安装与配置七、常用的控制类型八、常用的管理选项九、规则命令1.添加新规则2.查看规则列表3.设置默认策略4.删除规则5.清空规…...
第九章 Productions最佳实践 - Productions开发的最佳实践
文章目录 第九章 Productions最佳实践 - Productions开发的最佳实践Productions开发的最佳实践项目目标项目交付文档 第九章 Productions最佳实践 - Productions开发的最佳实践 Productions开发的最佳实践 本章是一个总体概述,旨在帮助团队成员为从事生产项目做好…...
RocketMQ 怎么实现的消息负载均衡以及怎么能够保证消息被顺序消费
一、RocketMQ 怎么实现的消息负载均衡 RocketMQ是一种开源的分布式消息中间件,它使用了一种称为消息负载均衡的机制来实现消息的分发和消费的负载均衡。RocketMQ的消息负载均衡主要是通过以下两个方面实现的: 消息队列分组(Message Queue G…...
【随笔记】全志 T507 PF4 引脚无法被正常设置为中断模式的问题分析
相关信息 硬件平台:全志T507 系统版本:Android 10 / Linux 4.9.170 问题描述:PF4 无法通过标准接口设置为中断模式,而 PF1、PF2、PF3、PF5 正常可用。 分析过程 一开始以为是引脚被其它驱动占用引起,或者该引脚不具…...
人手一个 Midjourney,StableStudio 重磅开源!
人手一个 Midjourney,StableStudio 重磅开源! Stability AI 公司在上个月 19 号推出了 Alpha 版本 StableLM 大语言模型,包含了 30 亿和 70 亿参数,并且支持商用。如今他们再次推出了 AI 图像生成平台 StableStudio,这…...
iptables防火墙(2)
iptables防火墙(2) 一、SNATSNAT应用环境SNAT原理SNAT转换前条件扩展 二、DNATDNAT应用环境DNAT原理DNAT转换前提条件扩展 三、防火墙规则的备份和还原导出(备份)所有表的规则导入(还原)规则 一、SNAT SNA…...
Windows和Kali上使用proxychains代理流量
Windows和Kali上使用proxychains代理流量 PS. 本文演示都是在kali进行的,如有出入还请联系我哦1. Linux(Debian)1.1. 检查一下是否有proxychains1.2 修改config文件 2. Linux(Debian)安装proxychians43. Windows3.1 下载3.2 配置 4. Windows下的配置5. 测试 PS. 写这…...
KEYSIGHT MSOS204A 2GHZ 4通道DSOS204A高清晰度示波器
KEYSIGHT是德DSOS204A/MSOS204A高清晰度示波器 附加功能: 2 GHz 带宽(可升级) 4 个模拟通道和 16 个数字通道 最大存储深度:800 Mpts(2 通道),400 Mpts(4 通道) 最大…...
最新Java适配商城系统
城前端功能展示 商城移动端 后端基于SpringBoot 研发,前端使用 Vue、uniapp开发 前后端分离,支持分布式部署,支持Docker,各个API独立,并且有独立的消费者 api不需要单独部署,只需启动一个jar包就可以正…...
【KVM虚拟化】· virsh管理命令
目录 🍁libvirt架构概述 🍁使用virsh管理虚拟机 🍂常用命令总结 🍁kvm基本功能管理 🍂帮助命令 🍂KVM的配置文件存放目录 🍂查看虚拟机状态 🍂虚拟机关机与开机 🍂强制虚…...
JS Es6中判断b数组对象是否有跟a数组对象相同的数值(例如:id),有的话就过滤掉
如下[数组]对象a和b let a[{id:1,value:this},{id:2,value:is}] let b[{id:1,value:hello},{id:3,value:world}]filter() 方法创建一个新的数组,新数组中的元素是通过检查指定数组中符合条件的所有元素。 some() 方法用于检测数组中的元素是否满足指定条件&#x…...
python获取某电商平台口红数据并制作词云
目录标题 前言开发环境:模块使用数据来源分析代码展示获取数据制作词云 尾语 💝 前言 嗨喽~大家好呀,这里是魔王呐 ❤ ~! 开发环境: Python 3.8 Pycharm 模块使用 requests jieba 结巴分词 wordcloud 词云 第三方模块安装: win R 输…...
阿里成立AIDC,用“增长”解题国际化
随着阿里巴巴集团2023财年年报的披露,AIDC也随即浮出了水面。 AIDC是阿里国际数字商业集团的英文简称,AIDC即Alibaba International Digital Commerce。阿里是在5月18日公布的截至2023年3月31日的2023财年Q4及全年财报,财报数据之外ÿ…...
openEuler23.09源码编译PostgreSQL16.3实战指南
1. 环境准备与系统配置 在openEuler23.09上编译PostgreSQL16.3之前,我们需要先做好基础环境配置。我建议使用全新安装的openEuler23.09系统,这样可以避免各种依赖冲突问题。通过执行cat /etc/os-release可以确认系统版本,输出应该包含VERSION…...
基于Transformer-GRU、Transformer、CNN-GRU、GRU、CNN五模型单变量时序预测一键对比
✅作者简介:热爱科研的Matlab仿真开发者,擅长毕业设计辅导、数学建模、数据处理、建模仿真、程序设计、完整代码获取、论文复现及科研仿真。👇 关注我领取海量matlab电子书和数学建模资料🍊个人信条:格物致知,完整Matl…...
Windows下TensorFlow CPU加速终极方案:AVX2指令集优化版安装指南
Windows平台TensorFlow CPU性能飞跃:AVX2指令集优化实战手册 当你在PyCharm中运行TensorFlow代码时,是否见过这样的警告:"Your CPU supports instructions that this TensorFlow binary was not compiled to use: AVX2"?…...
手把手教你搭建simple-breakpad-server在线解析服务(含curl上传示例)
构建企业级崩溃分析系统:从Simple-Breakpad-Server部署到实战解析 在软件开发的生命周期中,系统崩溃是无法完全避免的挑战。当用户报告"程序突然退出"或"闪退"时,传统的日志往往难以定位根本原因。这时,一个…...
GLM-4.1V-9B-Base实战:基于Visio流程图的企业智能审批系统设计与实现
GLM-4.1V-9B-Base实战:基于Visio流程图的企业智能审批系统设计与实现 1. 引言:当流程图遇上AI审批 最近帮一家制造企业做数字化转型时,遇到个有意思的挑战。他们的采购审批流程足足有12个环节,每次审批都要不同部门手动传递纸质…...
西门子博图编程:PLC状态机(二)ST语言实现并行状态机
1. 为什么需要并行状态机? 在PLC控制系统中,很多场景都需要处理多个同时发生的任务。比如一个包装生产线,可能需要同时监控传送带速度、检测产品位置、控制机械手动作。如果用传统的顺序状态机处理,程序会变得非常复杂且难以维护。…...
Youtu-VL-4B-Instruct快速上手:3个命令启动服务、5个API调用示例、10分钟掌握核心能力
Youtu-VL-4B-Instruct快速上手:3个命令启动服务、5个API调用示例、10分钟掌握核心能力 你是不是经常遇到这样的场景:拿到一张复杂的图表,想快速提取里面的数据;或者看到一张产品图,想知道里面有哪些东西、分别在哪里&…...
阿里云YUM源配置避坑指南
在CentOS 7上安装MySQL 8时,正确配置阿里云提供的YUM源是确保安装顺利、避免依赖冲突的关键。核心步骤包括清理系统旧有冲突软件包、配置稳定的软件源、处理GPG密钥验证问题。以下是一个结合官方实践和阿里云镜像优化的详细方案。 一、 核心步骤与对比 为了清晰展…...
保姆级教程!小程序开发只需3步,Gemini设计 + Trae开发 + 微信开发者工具预览上架
大家好,我是李奔腾。今天我想分享一下,如何通过AI工具快速设计和开发一个万年历小程序。借助 Gemini、Trae 和 微信开发者工具,几分钟时间就能让小程序顺利运行起来,极大地提升开发效率。第一步:使用Gemini设计小程序首…...
Vue3集成百度地图GL版:从自定义样式到动态轨迹绘制实战
1. Vue3集成百度地图GL版的前期准备 第一次在Vue3项目里用百度地图GL版时,我踩了不少坑。这里分享下最稳妥的集成方案,帮你避开那些我趟过的雷。首先得明白,百度地图GL版是基于WebGL技术的新一代地图API,相比传统版本性能更好、效…...
