springboot利用切面保存操作日志(支持Spring表达式语言(简称SpEL))

springboot利用切面保存操作日志(支持Spring表达式语言(简称SpEL))


文章目录

  • springboot利用切面保存操作日志(支持Spring表达式语言(简称SpEL))
  • 前言
  • 一、Spring EL是什么?
  • 二、使用步骤
    • 1.定义日志实体类LogRecord
    • 2.定义日志记录注解LogSnipper
    • 3.定义上下文容器SnipperContext
    • 4.实现切面
    • 5.定义日志模板解析器LogTplParser
    • 6.定义业务代码


前言

在注解中使用Spring EL表达式,切面解析实现自定义操作日志。


一、Spring EL是什么?

Spring表达式语言(简称SpEL)是一个支持查询并在运行时操纵一个对象图的功能强大的表达式语言。SpEL语言的语法类似于统一EL,但提供了更多的功能,最主要的是显式方法调用和基本字符串模板函数。参考:SpringEL 表达式语言(Spring Expression Language)

二、使用步骤

1.定义日志实体类LogRecord

/**
 * 日志实体类,属性对应{@link LogSnipper}注解中的属性
 */
@Data
@Builder
public class LogRecord {

    private String code;

    /**
     * 客户端ip
     */
    private String ipAddr;

    /**
     * 业务编号(可以是任何标识)
     */
    private String bizNo;

    /**
     * 对应{@link LogSnipper}注解中的success或者fail,当方法成功调用时,获取success中的值,反之则获取fail中的值
     */
    private String content;

    /**
     * 日志类型
     */
    private String category;

    /**
     * 操作者
     */
    private String operator;

    /**
     * 备用字段,当其他字段不够用时可以使用该字段
     */
    private String addition;

    /**
     * 操作时间
     */
    private Date createTime;

}

2.定义日志记录注解LogSnipper

@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
@Inherited
@Documented
public @interface LogSnipper {

    /**
     * 操作人
     * @return
     */
    String operator() default "";

    /**
     * 操作成功时定义的模板
     * @return
     */
    String success();

    /**
     * 操作失败时定义的模板
     * @return
     */
    String fail() default "";

    /**
     * 业务主键
     * @return
     */
    String bizNo() default "";

    /**
     * 日志类型
     * @return
     */
    String category();

    /**
     * 其他信息,用于保存其他额外信息
     *
     * @return
     */
    String addition() default "";



}

3.定义上下文容器SnipperContext

/**
 * 上下文容器,用来存放线程处理过程中的信息
 * <p>
 */
public class SnipperContext {

    private static ThreadLocal<Map<String, Object>> context = new ThreadLocal<>();

    static {
        context.set(new HashMap<>());
    }

    public static Map<String, Object> getContext() {
        return context.get();
    }

    public static void set(String key, Object val) {
        Map<String, Object> map = new HashMap<>();
        map.put(key, val);
        context.set(map);
    }

    public static Object get(String key) {
        return context.get().get(key);
    }

    public static void clear() {
        context.remove();
    }

}

4.实现切面

/**
 * 日志记录切面
 *
 */
@Aspect
public class LogAspect {

    @Autowired
    private LogTplParser logTplParser;

    @Autowired(required = false)
    private LogPersistent logPersistent;

    @Pointcut("@annotation(com.sztech.logsnipper.LogSnipper)")
    public void snip(){}

    /**
     * 正常执行方法,执行success模板
     */
    @AfterReturning(pointcut = "snip()", returning = "returnValue")
    public void doAfter(JoinPoint joinPoint, Object returnValue) {
        //获取注解的实例
        LogSnipper logSnipper = getLogSnipper(joinPoint);
        String success = logSnipper.success();
        //表达式上下文,将方法中的参数设置到spel容器中
        EvaluationContext evaluationContext = generateEvaluationContext(joinPoint);
        /**
         * 将方法的返回结果设置进内置变量中
         * 其中_rs_对应为返回值
         * _ex_对应异常原因
         * _context_对应上下文内容,为map类型
         *
         */
        evaluationContext.setVariable(LogTplParser._RS_, returnValue);
        evaluationContext.setVariable(LogTplParser._CONTEXT, SnipperContext.getContext());
        saveLog(logSnipper, success, evaluationContext);
    }

    /**
     * 非正常执行,执行fail模板
     */
    @AfterThrowing(pointcut = "snip()", throwing = "exception")
    public void doFail(JoinPoint joinPoint, Exception exception) {
        LogSnipper logSnipper = getLogSnipper(joinPoint);
        String fail = logSnipper.fail();
        /**
         * 将方法的返回结果设置进内置变量中
         * 其中_rs_对应为返回值
         * _ex_对应异常原因
         * _context_对应上下文内容,为map类型
         *
         */
        EvaluationContext evaluationContext = generateEvaluationContext(joinPoint);
        evaluationContext.setVariable(LogTplParser._EX_, exception);
        evaluationContext.setVariable(LogTplParser._CONTEXT, SnipperContext.getContext());
        saveLog(logSnipper, fail, evaluationContext);
    }

    /**
     * 保存日志
     * @param logSnipper
     * @param tpl
     * @param evaluationContext
     */
    private void saveLog(LogSnipper logSnipper, String tpl, EvaluationContext evaluationContext) {
        String content = logTplParser.parseTpl(tpl, evaluationContext);
        String operator = logTplParser.parseTpl(logSnipper.operator(), evaluationContext);
        String bizNo = logTplParser.parseTpl(logSnipper.bizNo(), evaluationContext);
        String category = logTplParser.parseTpl(logSnipper.category(), evaluationContext);
        String addition = logTplParser.parseTpl(logSnipper.addition(), evaluationContext);
        LogRecord logRecord =LogRecord.builder()
                .content(content)
                .code(IdUtil.simpleUUID())
                .operator(operator)
                .bizNo(bizNo)
                .category(category)
                .addition(addition)
                .createTime(new Date())
                .build();
        if(null != logPersistent) {
        	//实现日志存储
            logPersistent.save(logRecord);
        }
    }

    /**
     * 将方法中的参数设置到spel容器中
     *
     * @param joinPoint
     * @return
     */
    private EvaluationContext generateEvaluationContext(JoinPoint joinPoint) {
        EvaluationContext evaluationContext = new StandardEvaluationContext();
        //获取方法参数值
        Object[] args = joinPoint.getArgs();
        //获取方法参数名
        String[] parameterNames = ((MethodSignature) joinPoint.getSignature()).getParameterNames();
        int i = 0;
        for(Object arg : args) {
            evaluationContext.setVariable(parameterNames[i++], arg);
        }
        return evaluationContext;
    }

    /**
     * 获取方法{@link LogSnipper}的注解实例
     * @param joinPoint
     * @return
     */
    private LogSnipper getLogSnipper(JoinPoint joinPoint) {
        MethodSignature signature = (MethodSignature) joinPoint.getSignature();
        LogSnipper logSnipper = signature.getMethod().getAnnotation(LogSnipper.class);
        return logSnipper;
    }

    /**
     * 最后执行,用于释放资源
     */
    @After("snip()")
    public void finish() {
        //释放上下文
        SnipperContext.clear();
    }
}

5.定义日志模板解析器LogTplParser

/**
 * 日志模板解析器
 *
 */
@Component
public class LogTplParser {

    private ExpressionParser expressionParser = new SpelExpressionParser();

    /**
     * 通配符正则表达式:{#user.name},user为实例对象,name为对象属性,也可以使用getName()方法
     */
    private static final String expressionRegex = "\\{#.*?}";

    public static final String _CONTEXT = "_context_";

    public static final String _EX_ = "_ex_";

    public static final String _RS_ = "_rs_";

    private static final Pattern pattern = Pattern.compile(expressionRegex);

    /**
     * 解析Spel模板字符串
     *
     * @param tpl 模板字符串,例如:"操作人:{#username}, 操作模块:{#module.name}"等,使用了spring的spel表达式
     * @param context 对象容器,将spel表达式对应的值或者对象呢set进容器中
     * @return
     */
    public String parseTpl(String tpl, EvaluationContext context) {
        Matcher matcher = pattern.matcher(tpl);
        List<String> sPelList = new ArrayList<>();
        //匹配正则表达式,可能会出现匹配到多个
        while (matcher.find()) {
            sPelList.add(matcher.group());
        }
        //如果没有匹配到通配符,直接返回原字符串
        if(sPelList.size() == 0) {
            return tpl;
        }
        String[] datas = new String[sPelList.size()];
        int i = 0;
        //从context中取出通配符对应的值
        for(String sp : sPelList) {
            Expression expression = expressionParser.parseExpression(sp);
            datas[i++] = expression.getValue(context, String.class);
        }
        //格式化字符串,并将通配符的值填充到字符串中
        String formatStr = tpl.replaceAll(expressionRegex, "%s");
        return String.format(formatStr, datas);
    }

}

6.定义业务代码

@RestController
public class AController {

	@Autowired
    private AService aService;

	@GetMapping("/sayBye")
    public String sayBye(String msg) {
        User user = new User();
        user.setName("李白");
        user.setAge(12);
        user.setDepartment("001");
        SnipperContext.set("currentUser", user);
        SnipperContext.set("nickName", "汪沦");
        return aService.sayBye(msg, user);
    }
    }

@Component
public class aService {
 	@LogSnipper(category = "打招呼行为",
            success = "{#_context_.get(\"nickName\")}向{#user.name}说:{#msg}",
            fail = "{#user.name}有事,请下次再来",
            bizNo = "{#user.department}",
            operator = "{#_context_.get(\"nickName\")}")
    public String sayBye(String msg, User user) {
        return "sayBye";
    }
}


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

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

相关文章

CTF-密码学基础

概述 密码学(Cryptolopy)&#xff1a;是研究信息系统安全保密的科学 密码学研究的两个方向&#xff1a; 密码编码学(Cryptography)&#xff1a;主要研究对信息进行编码&#xff0c;实现对信息的隐蔽密码分析学(Cryptanalytics)&#xff1a;主要研究加密信息的破译或消息的伪造…

Baidu Comate——让软件研发更高效、更智能

个人名片&#xff1a; &#x1f60a;作者简介&#xff1a;一名大二在校生 &#x1f921; 个人主页&#xff1a;坠入暮云间x &#x1f43c;座右铭&#xff1a;给自己一个梦想&#xff0c;给世界一个惊喜。 &#x1f385;**学习目标: 坚持每一次的学习打卡 文章目录 一、Baidu Co…

Spring 事务及事务传播机制(1)

目录 事务 回顾: 什么是事务 为什么需要事务 事务的操作 Spring事务的实现 Spring编程式事务(简单了解即可, 问就是基本不用) 观察事务提交 观察事务回滚 Spring声明式事务 Transactional Transactional作用 事务 回顾: 什么是事务 定义: 事务是指逻辑上的一组操作, 构…

最大数字——蓝桥杯十三届2022国赛大学B组真题

问题分析 这道题属于贪心加回溯。所有操作如果能使得高位的数字变大必定优先用在高位&#xff0c;因为对高位的影响永远大于对低位的影响。然后我们再来分析一下&#xff0c;如何使用这两种操作&#xff1f;对于加操作&#xff0c;如果能使这一位的数字加到9则变成9&#xff0…

^_^填坑备忘^_^C#自动化编程实现STK+Exata对卫星互联网星座进行网络仿真

C#实际选择 STK11版本 or STK12版本的问题备注。 【C#自动化客户端调用STK时&#xff0c;实际选择 STK11版本 or STK12版本 的调试运行备注】 以下代码“更新并重新打包备份为”〔testSTKQualNetInterface备份08.1_★避坑★【种子卫星&#xff1a;天线直接安装在卫星上&#…

电机控制系列模块解析(19)—— 反电势观测器

随着现代工业自动化技术的飞速发展&#xff0c;交流电机作为关键的动力装置&#xff0c;其控制精度与效率日益受到重视。其中&#xff0c;无位置传感器控制技术由于其成本低、可靠性高、系统简洁等优点&#xff0c;逐渐成为研究热点。本文将对交流电机反电势观测器这一关键技术…

三维空间刚体运动

三维空间刚体运动是指刚体在三维空间中的运动&#xff0c;这种运动由平移和旋转构成。平移是指物体在空间中沿某一方向移动一定的距离&#xff0c;而旋转则是指物体绕某一轴旋转一定的角度。这两种运动都不会改变物体的形状和大小&#xff0c;因此被称为刚体运动。 在描述三维…

Qt跨平台开发demo(适用萌新)

最近需要参与一款Qt跨平台的软件开发&#xff0c;在此之前&#xff0c;特把基础信息做学习和梳理&#xff0c;仅供参考。 所使用的技术和版本情况如下&#xff1a; 虚拟机&#xff1a;VMware 16.2.5操作系统&#xff1a;ubuntu-20.04.6-desktop-amd64&#xff1a;Mysql数据库…

大模型入坑记:搭建本地大模型微调环境

为了让大模型发挥更大用途&#xff0c;决定在本地搭建大模型微调环境&#xff0c;在原有的PC上加装Tesla V100&#xff0c;前前后后耗时一个多月&#xff0c;遇到若干技术问题&#xff0c;好在目前已基本得到解决&#xff0c;也打破了很多网上店家包括身边专家对GPU搭建上的一些…

正版软件 | Total Uninstall - Windows 全功能卸载程序 新手入门教程

『软件简介』 Total Uninstall 是一款先进的系统监控与卸载工具&#xff0c;它通过创建安装前后的系统快照&#xff0c;为用户提供了一种全新的程序管理方式。这款软件具备两个主要功能&#xff1a;一是能够独立于系统自带的卸载程序&#xff0c;彻底移除已安装的应用程序&…

FPGA -手写异步FIFO

一&#xff0c;FIFO原理 FIFO&#xff08;First In First Out&#xff09;是一种先进先出的数据缓存器&#xff0c;没有外部读写地址线&#xff0c;使用起来非常简单&#xff0c;只能顺序写入数据&#xff0c;顺序的读出数据&#xff0c;其数据地址由内部读写指针自动加1完成&a…

开源数据可视化大屏对接表单数据实践!

如果你需要一个表单系统&#xff0c;进行数据收集&#xff1b;可以使用tduck填鸭进行私有化部署&#xff0c;进行表单制作&#xff0c;完成数据收集。 在实际业务中&#xff0c;往往需要将收集的数据进行展示或分析&#xff1b;此时就可以使用表单数据推送到TReport中&#xf…

AMBA总线介绍

AMBA&#xff08;Advanced Microcontroller Bus Architecture&#xff09;是由ARM&#xff08;Advanced RISC Machines&#xff09;公司设计的一种高性能、高带宽的总线架构。AMBA总线广泛应用于各种嵌入式系统中&#xff0c;包括数字信号处理器、图形处理器、嵌入式处理器以及…

收银系统源码--什么是千呼智慧新零售系统?

千呼智慧新零售系统是一套针对零售行业线上线下一体化收银系统。给门店提供线下称重收银、o2o线上商城、erp进销存、精细化会员管理、丰富营销插件等一体化解决方案。多端数据打通&#xff0c;实现线上线下一体化&#xff0c;提升门店工作效率&#xff0c;实现数字化升级&#…

vue3 依赖-组件tablepage-vue3版本1.0.3更新内容

github求⭐ 可通过github 地址和npm 地址查看全部内容 vue3 依赖-组件tablepage-vue3说明文档&#xff0c;列表页快速开发&#xff0c;使用思路及范例-汇总 vue3 依赖-组件tablepage-vue3说明文档&#xff0c;列表页快速开发&#xff0c;使用思路及范例&#xff08;Ⅰ&#…

【第38天】SQL进阶-SQL设计优化-范式设计(SQL 小虚竹)

回城传送–》《100天精通MYSQL从入门到就业》 文章目录 零、前言一、练习题目二、SQL思路初始化数据什么是范式设计第一范式&#xff08;1NF&#xff09;第二范式&#xff08;2NF&#xff09;第三范式&#xff08;3NF&#xff09; 三、总结四、参考 零、前言 今天是学习 SQL …

期权和期货有什么区别?

今天期权懂带你了解期权和期货有什么区别&#xff1f;期权和期货是两种常见的衍生金融工具&#xff0c;它们在结构和盈利方式上存在一些关键的区别&#xff1a; 期权 期权是一种给予持有者在未来某个时间以特定价格买入或卖出基础资产的权利&#xff0c;但不是义务。期权的主要…

ORACLE 19C RAC DIAG进程消耗大量内存的分析

近期一个ORACLE 19C的RAC环境&#xff0c;多次出现数据库实例的后台进程DIAG消耗很多内存&#xff08;达到20G&#xff09;&#xff0c;节点1、节点2都出现过次问题。 问题分析&#xff1a;通过对DIAG进程TRACE分析&#xff0c;结合在ORACLE官方后台进行问题、BUG查询匹配&…

uniapp怎么使用jsx

安装vitejs/plugin-vue-jsx npm install vitejs/plugin-vue-jsx -Dvite.config.js配置 import { defineConfig } from "vite"; import uni from "dcloudio/vite-plugin-uni"; import vueJsx from vitejs/plugin-vue-jsxexport default defineConfig({plu…

初始Java篇(JavaSE基础语法)(7)抽象类和接口(下)

找往期文章包括但不限于本期文章中不懂的知识点&#xff1a; 个人主页&#xff1a;我要学编程(ಥ_ಥ)-CSDN博客 所属专栏&#xff1a;JavaSE 接上文&#xff1a;初始Java篇&#xff08;JavaSE基础语法&#xff09;&#xff08;7&#xff09;抽象类和接口&#xff08;上&#xf…
最新文章