网站设计两边为什么要留白,网页制作基础教程第2版答案,教育机构培训,wordpress怎么破解springboot利用切面保存操作日志#xff08;支持Spring表达式语言#xff08;简称SpEL#xff09;#xff09; 文章目录 springboot利用切面保存操作日志#xff08;支持Spring表达式语言#xff08;简称SpEL#xff09;#xff09;前言一、Spring EL是什么#xff1f…springboot利用切面保存操作日志支持Spring表达式语言简称SpEL 文章目录 springboot利用切面保存操作日志支持Spring表达式语言简称SpEL前言一、Spring EL是什么二、使用步骤1.定义日志实体类LogRecord2.定义日志记录注解LogSnipper3.定义上下文容器SnipperContext4.实现切面5.定义日志模板解析器LogTplParser6.定义业务代码 前言
在注解中使用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 ThreadLocalMapString, Object context new ThreadLocal();static {context.set(new HashMap());}public static MapString, Object getContext() {return context.get();}public static void set(String key, Object val) {MapString, 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 {Autowiredprivate 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);ListString 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 {Autowiredprivate 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;}
}