java面向切面编程AOP案例,记录请求日志

news/2024/4/26 0:58:20/文章来源:https://blog.csdn.net/qq_30270931/article/details/128101539

提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档

文章目录

  • 前言
  • 一、aop是什么?
  • 二、使用步骤
    • 1.引入pom文件依赖
    • 2.请求日志记录
  • 总结


前言

本文主要描述面向切面编程AOP的用例

一、AOP是什么?

利用AOP可以对业务逻辑的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发的效率

二、使用步骤

1.引入pom文件依赖

代码如下(示例):

<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-aop</artifactId><version>2.6.4</version>
</dependency>

2.请求日志记录

代码如下(示例):

@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.TYPE, ElementType.METHOD})
public @interface OptLog {/*** 操作日志类型** @return*/OperationTypeEnum scenesType();/*** 请求处理状态** @return*/int status() default 0;/*** 写SpEL表达式 获取request请求内容** @return*/String request() default "";/*** request请求内容的类型** @return*/Class<?> requestType() default String.class;/*** 默认 系统管理员* 其他的取当前登录人** @return*/String operationUser() default "系统管理员";/*** 默认 系统管理员* 其他的取当前登录人** @return*/long operationUserId() default 0L;
}
@Slf4j
@Aspect
@Component
public class OptLogAspect {/*** 用于SpEL表达式解析.*/private final SpelExpressionParser spelExpressionParser = new SpelExpressionParser();/*** 用于获取方法参数定义名字.*/private final DefaultParameterNameDiscoverer nameDiscoverer = new DefaultParameterNameDiscoverer();@Autowiredprivate InLogRepository inLogRepository;@Pointcut("@annotation(com.yssk.interfaces.biz.annotation.OptLog)")public void optLogAspect() {}@Around("optLogAspect()")public Object doAround(ProceedingJoinPoint joinPoint) {Object obj = null;try {MethodSignature methodSignature = (MethodSignature) joinPoint.getSignature();//获取方法参数值Object[] args = joinPoint.getArgs();OptLog optLog = getTargetAnnotation(joinPoint);String request = optLog.request();Class<?> requestType = optLog.requestType();// 获取request请求内容Object requestKeyVal = getValBySpEl(request, methodSignature, args, requestType);if(Objects.nonNull(requestKeyVal)){Long id = handleOptRecordBefore(requestKeyVal);log.info("rizhi:{}", id);obj = joinPoint.proceed();handleOptRecordAfter(id, obj);}} catch (Throwable e) {log.warn("记录请求日志出现异常...", e);}return obj;}private Long handleOptRecordBefore(Object request){log.info("请求日志保存");String jsonStr = JSONObject.toJSONString(request);String requestUrl = "";if(ObjectUtils.isNotEmpty(request)){if(request instanceof RequestDTO){requestUrl = ((RequestDTO<?>) request).getRequestUrl();}}// 保存请求日志InLog inLog = new InLog();inLog.setRequest(jsonStr);inLog.setRequestUrl(requestUrl);inLog.setStatus(0);inLogRepository.save(inLog);return inLog.getId();}private void handleOptRecordAfter(Long id, Object response){log.info("请求日志修改");String jsonStr = JSONObject.toJSONString(response);int status = 0;if(ObjectUtils.isNotEmpty(response)){if(response instanceof ResponseDTO){if(((ResponseDTO<?>) response).getError_code() == 0){status = 1;}}}// 修改请求日志LambdaUpdateWrapper<InLog> updateWrapper = new LambdaUpdateWrapper<>();updateWrapper.set(InLog::getStatus, status);updateWrapper.eq(InLog::getId, id);updateWrapper.set(InLog::getResponse, jsonStr);inLogRepository.update(updateWrapper);}/*** 获取注解* @param point* @return*/public static OptLog getTargetAnnotation(JoinPoint point) {try {OptLog annotation = null;if (point.getSignature() instanceof MethodSignature) {Method method = ((MethodSignature) point.getSignature()).getMethod();if (method != null) {annotation = method.getAnnotation(OptLog.class);}}return annotation;} catch (Exception e) {log.warn("获取 {}.{} 的 @OptLog 注解失败", point.getSignature().getDeclaringTypeName(), point.getSignature().getName(), e);return null;}}/*** 解析spEL表达式*/private Object getValBySpEl(String spEl, MethodSignature methodSignature, Object[] args, Class<?> classType) {try {//获取方法形参名数组String[] paramNames = nameDiscoverer.getParameterNames(methodSignature.getMethod());if (paramNames != null && paramNames.length > 0) {Expression expression = spelExpressionParser.parseExpression(spEl);// spring的表达式上下文对象EvaluationContext context = new StandardEvaluationContext();// 给上下文赋值for (int i = 0; i < args.length; i++) {context.setVariable(paramNames[i], args[i]);context.setVariable("p" + i, args[i]);}return expression.getValue(context, classType);}} catch (Exception e) {log.warn("解析请求日志的el表达式出错", e);}return null;}
}

总结


以上就是今天要讲的内容,本文仅仅简单介绍了aop的使用法。

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

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

相关文章

供应荧光染料AF532 活性酯,AF532-NHS,CAS:477876-64-5

一&#xff1a;产品描述 1、名称 AF5 532酯 AF532-NHS AF532 活性酯 Alexa Fluor 532 AF532 NHS ester 2、CAS编号&#xff1a;477876-64-5 3、分子式&#xff1a;C34H33N3O11S2 4、分子量&#xff1a;723.77 5、质量控制&#xff1a;95% 6、储存&#xff1a; -20…

Arduino开发实例-DIY酒精浓度检测计

DIY酒精浓度检测计 在本文中,将详细介绍如何创建一个简单的酒精检测器。 它可以在各种应用领域中使用。市场上有许多先进的酒精传感器,价格合理,但我们在这里使用一些基本的微控制器来制作这个项目,如 Arduino、LED、蜂鸣器和 MQ3 酒精传感器。 1、MQ-3传感器介绍 MQ-3传…

A*算法-Python实现

好久没有在CSDN上发文章了&#xff0c;快一年了吧。这两天重新登录了一下&#xff0c;不看不知道&#xff0c;一看吓一跳&#xff0c;没想到访问量快13万了。 之前写博客的时候&#xff0c;想着把一些有用的东西写下来&#xff0c;一方面是当做笔记了&#xff0c;免得以后忘记…

5G无线技术基础自学系列 | SU-MIMO原理

素材来源&#xff1a;《5G无线网络规划与优化》 一边学习一边整理内容&#xff0c;并与大家分享&#xff0c;侵权即删&#xff0c;谢谢支持&#xff01; 附上汇总贴&#xff1a;5G无线技术基础自学系列 | 汇总_COCOgsta的博客-CSDN博客 通过多天线技术支持单用户在上下行数据…

Vue3框架中路由的使用和局部刷新的功能(第十一课)

使用vue-router的步骤:p第一步&#xff1a;创建路由需要映射的组件&#xff08;打算显示的页面&#xff09;&#xff1b;p第二步&#xff1a;通过createRouter创建路由对象&#xff0c;并且传入routes和history模式&#xff1b;配置路由映射: 组件和路径映射关系的routes数组&a…

DFL3:软件版本的选择和安装详解

这本是一个简单的问题&#xff0c;但是对于新手而言&#xff0c;所有问题&#xff0c;总是说的越清楚越仔细越好。我之所以这么说&#xff0c;肯定是有人问了。所以我就专门开一篇文章来说一说&#xff0c;软件版本的异同&#xff0c;以及如何选择。针对不同的语言&#xff0c;…

mysql相关基础知识篇(五)

1.MySQL 事务的四大特性说一下&#xff1f; 原子性&#xff1a;事务作为一个整体被执行&#xff0c;包含在其中的对数据库的操作要么全部被执行&#xff0c;要么都不执行。一致性&#xff1a;指在事务开始之前和事务结束以后&#xff0c;数据不会被破坏&#xff0c;假如 A 账户…

(2)点云库处理学习——剔除点云值

1、主要参考 1.1参考地址 (1) 点云离群点剔除 — open3d python_Coding的叶子的博客-CSDN博客_离群点去除 (2) open3d之点云异常值去除&#xff08;笔记5&#xff09;_Satellite_H的博客-CSDN博客 (3)斯坦福经典兔子的点云数据下载地址 下载地址 Model : Bunny 1.2兔子…

Git 打patch (打补丁)的使用

patch 的使用 一般是diff ,apply ,format-patch,am 1 生成patch git diff > test.patch 这个是打补丁(test.patch自己取的名字,这个命令可以看出没有指定修改的问题所以默认把所有修改的文件都打patch了,同时还需要注意,这里是本地修改的没有执行add缓存的) 如果想指定某…

[附源码]计算机毕业设计SpringBoot高血压分析平台

项目运行 环境配置&#xff1a; Jdk1.8 Tomcat7.0 Mysql HBuilderX&#xff08;Webstorm也行&#xff09; Eclispe&#xff08;IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持&#xff09;。 项目技术&#xff1a; SSM mybatis Maven Vue 等等组成&#xff0c;B/S模式 M…

时序特征提取工具

在选择了需要提取的特征&#xff0c;确定了时序数据特征提取数据集的长度并对先验知识建模之后&#xff0c;就需要利用工具搭建特征提取系统。科研机构围绕不同问题域搭建的开源时序数据特征提取工具已经不少&#xff0c;我们可以利用这些工具快速实现希望达成的算法效果。下面…

驱动——platform驱动总线三种匹配方式

三种platform驱动匹配方式代码案例以及现象 方式一&#xff1a;通过设置名字进行匹配 相关API简介&#xff1a; 1、platform_device的API ①分配对象 struct platform_device { const char *name;//用于进行匹配的名字 int id;//总线号 PLATFORM_DEVID_AUTO&#xff08;自…

ARM cortex-A7核UART实验 收发数据

头文件&#xff1a; 1 #ifndef __UART4_H__ 2 #define __UART4_H__ 3 4 #include "../common/include/stm32mp1xx_rcc.h" 5 #include "..…

【Android App】获取照片里的位置信息及使用全球卫星导航系统(GNSS)获取位置实战(附源码和演示 超详细)

需要全部代码请点赞关注收藏后评论区留言私信~~~ 一、获取照片里的位置信息 手机拍摄的相片还保存着时间、地点、镜头参数等信息&#xff0c;这些信息由相片接口工具ExifInterface管理&#xff0c;它的常用方法说明如下&#xff1a; getLatLong&#xff1a;获取相片拍摄时候的…

【人工智能 机器学习 深度学习】基础选择题1~30题 练习

目录 一、1~10题1.1 题目1.2 答案二、11~20题2.1 题目2.2 答案三、21~30题3.1 题目3.2 答案写在前面:适用于对 人工智能&机器学习&深度学习 进行复习的同学,同时,也可以通过基础题目的练习,加深理解。 一、1~10题 均是先给出10道题目,而后给出 10道题目的答案。 …

Python用广义加性模型GAM进行时间序列分析

每当你发现一个与时间对应的趋势时&#xff0c;你就会看到一个时间序列。我们围绕广义加性模型GAM技术进行一些咨询&#xff0c;帮助客户解决独特的业务问题。研究金融市场表现和天气预报的事实上的选择&#xff0c;时间序列是最普遍的分析技术之一&#xff0c;因为它与时间有着…

关于TreeView的简单使用(Qt6.4.1)

前言 TreeView是在Qt6.3中加入的&#xff0c;弥补了Qt中无官方树图。笔者上手尝试了下&#xff0c;虽然有点麻烦&#xff0c;但官方也做了不少简化。 本次教程&#xff0c;笔者创建一个简单的示例&#xff0c;以帮助读者使用TreeView。 一、创建模型类 当前模型需要使用C定义…

婚纱预订小程序开发,商家线上展示平台

婚纱代表着纯洁与忠贞&#xff0c;也是爱情永恒的见证者&#xff0c;穿上洁白的婚纱嫁给自己心爱的人是每个女生的梦想&#xff0c;婚纱对于每一个女生来说都有着重要的意义&#xff0c;所以选择一件美丽且适合的婚纱非常重要&#xff0c;因此人们在选择婚纱时会花费很多的时间…

Web3中文|区块链游戏的成长之痛

来源 | cointelegraph 编译 | DaliiNFTnews.com 在过去十年中&#xff0c;手机游戏已成为互动娱乐产业的重要支柱&#xff0c;得益于智能手机的普及&#xff0c;来自世界各地的用户都成为了硬核游戏玩家。 现在&#xff0c;区块链技术的出现正在推动一种范式的转变&#xff…

KNN最近邻算法分析及实现(Python实现)

KNN最近邻算法分析及实现&#xff08;代码附录后文&#xff09;1 KNN算法简介2 KNN基本原理3 简单实现KNN分析代码附录(Python)&#xff1a;呆&#xff0c;站住别跑&#xff0c;留个赞&#xff0c;给个关注嘛都看到这了Author&#xff1a; Nirvana Of Phoenixl Proverbs for yo…