简介

创建自定义日志注解,对相关接口记录请求日志。

环境

SpringBoot

实现

注解定义

定义注解类

package com.zk.app.annotation;

import com.zk.app.enums.UserLogTypeEnum;

import java.lang.annotation.*;

/**
* @program: ZK
* @description: 操作日志注解
* @author: zk
* @create: 2024-07-16 15:42
**/
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface OperationLog {
String operation() default ""; UserLogTypeEnum operationType() default UserLogTypeEnum.OTHER; boolean isSaveIp() default true; boolean isSavePath() default true; boolean isSaveRequestData() default true; boolean isSaveUserId() default true;
}

@Target@Retention是元注解,用来指定自定义注解的作用目标(比如类、方法)及其生命周期(源码级、编译期或运行时)。

ElementType.METHOD 指定注解可以应用于方法

RetentionPolicy.RUNTIME 注解在运行时可用

UserLogTypeEnum日志类型

package com.zk.app.enums;

/**
* @program: ZK
* @description: 用户操作日志类型
* @author: zk
* @create: 2024-07-16 14:36
**/
public enum UserLogTypeEnum {
/**
* 登录
*/
LOGIN("登录"),
/**
* 登出
*/
LOGOUT("登出"),
/**
* 注册
*/
REGISTER("注册"),
/**
* 查询
*/
SELECT("查询"),
/**
* 插入
*/
INSERT("插入"),
/**
* 更新
*/
UPDATE("更新"),
/**
* 删除
*/
DELETE("删除"),
/**
* 其他
*/
OTHER("其他"); private String desc; UserLogTypeEnum(String desc) {
this.desc = desc;
} public String getDesc() {
return desc;
}
}

添加依赖

添加aop依赖

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

定义切面

定义切面

package com.zk.app.aop;

import com.zk.app.annotation.OperationLog;
import com.zk.app.constants.StringConstant;
import com.zk.app.entity.UserLog;
import com.zk.app.service.UserLogService;
import com.zk.app.utils.IpUtil;
import com.zk.app.utils.UserThreadLocalUtil;
import org.apache.commons.lang3.StringUtils;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Pointcut;
import org.aspectj.lang.reflect.MethodSignature;
import org.springframework.beans.factory.annotation.Autowired;
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.lang.reflect.Method;
import java.util.Objects; /**
* @program: ZK
* @description: 操作日志aop
* @author: zk
* @create: 2024-07-16 16:52
**/
@Aspect
@Component
public class OperationLogAop {
@Autowired
private UserLogService userLogService; @Pointcut("@annotation(com.zk.app.annotation.OperationLog)")
public void operationLogCheck() {
} @Around("operationLogCheck()")
public Object doAround(ProceedingJoinPoint joinPoint) throws Throwable {
MethodSignature methodSignature = (MethodSignature) joinPoint.getSignature();
Method method =methodSignature.getMethod();
OperationLog operationLog = method.getAnnotation(OperationLog.class);
Object proceed = joinPoint.proceed();
if(Objects.isNull(operationLog)){
return proceed;
}
UserLog userLog = new UserLog();
if(operationLog.isSaveIp() || operationLog.isSaveRequestData()){
// 获取HttpServletRequest对象
ServletRequestAttributes attributes = (ServletRequestAttributes) RequestContextHolder.getRequestAttributes();
HttpServletRequest request = attributes.getRequest();
if(operationLog.isSaveIp()){
userLog.setLogIp(IpUtil.getIpAddr(request));
}
if(operationLog.isSaveRequestData()){
userLog.setLogContent(StringUtils.join(joinPoint.getArgs(), StringConstant.COMMA));
}
}
userLog.setLogTitle(operationLog.operation());
userLog.setLogType(operationLog.operationType());
if(operationLog.isSavePath()){
userLog.setLogPath(method.getDeclaringClass().getName() + StringConstant.HASH + method.getName());
}
if(operationLog.isSaveUserId()){
userLog.setUserId(UserThreadLocalUtil.getUserId());
}
userLogService.saveUserLog(userLog);
return proceed;
}
}

@Around注解用于指定环绕通知,即在方法执行前后都进行操作

日志入库

这里将操作记录及入参拆为主子表

@Transactional
public UserLog saveUserLog(UserLog userLog) {
userLogMapper.insert(userLog);
if(userLog.getId()>0 && Objects.nonNull(userLog.getLogContent())){
UserLogExt userLogExt = new UserLogExt();
Date userLogCreateTime = userLogMapper.selectById(userLog.getId()).getCreateTime();
userLogExt.setLogId(userLog.getId());
userLogExt.setCreateDate(userLogCreateTime);
userLogExt.setLogContent(StringUtil.textStringCheck(userLog.getLogContent()));
userLogExtService.save(userLogExt);
}
return userLog;
}

数据表

操作日志表


CREATE TABLE `zk_user_log` (
`id` bigint NOT NULL AUTO_INCREMENT COMMENT '自增id',
`user_id` bigint NOT NULL DEFAULT '0' COMMENT '操作用户',
`log_ip` varchar(255) NOT NULL DEFAULT '' COMMENT '操作ip',
`log_path` varchar(255) NOT NULL DEFAULT '' COMMENT '操作路径',
`log_type` varchar(255) NOT NULL COMMENT '日志类型',
`log_title` varchar(255) NOT NULL COMMENT '日志描述',
`log_status` int NOT NULL DEFAULT '0' COMMENT '日志状态:0正常;1禁用',
`create_time` datetime NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '创建时间',
`create_user_id` bigint NOT NULL DEFAULT '0' COMMENT '创建人',
`update_time` datetime NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP COMMENT '修改时间',
`update_user_id` bigint NOT NULL DEFAULT '0' COMMENT '修改人',
`is_delete` tinyint(1) NOT NULL DEFAULT '0' COMMENT '是否删除:0否;1是',
`remarks` varchar(255) NOT NULL DEFAULT '' COMMENT '备注',
`version` bigint NOT NULL DEFAULT '0' COMMENT '版本号',
PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_ai_ci COMMENT='操作日志表';

操作日志扩展表

按日期进行分区,例如及时清理及减少碎片率

分区键以主表创建日期,便于跟进分区键查询


CREATE TABLE `zk_user_log_ext` (
`id` bigint NOT NULL AUTO_INCREMENT COMMENT '自增id',
`log_id` bigint NOT NULL COMMENT '操作日志id',
`log_content` mediumtext NOT NULL COMMENT '日志详细信息',
`create_time` datetime NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '创建时间',
`create_date` date NOT NULL COMMENT '创建日期',
`create_user_id` bigint NOT NULL DEFAULT '0' COMMENT '创建人',
`update_time` datetime NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP COMMENT '修改时间',
`update_user_id` bigint NOT NULL DEFAULT '0' COMMENT '修改人',
`is_delete` tinyint(1) NOT NULL DEFAULT '0' COMMENT '是否删除:0否;1是',
`remarks` varchar(255) NOT NULL DEFAULT '' COMMENT '备注',
`version` bigint NOT NULL DEFAULT '0' COMMENT '版本号',
PRIMARY KEY (`id`,`create_date`)
) ENGINE=InnoDB AUTO_INCREMENT=12 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_ai_ci COMMENT='操作日志扩展表'
/*!50500 PARTITION BY RANGE COLUMNS(create_date)
(PARTITION p20240722 VALUES LESS THAN ('2024-07-23') ENGINE = InnoDB,
PARTITION p20240723 VALUES LESS THAN ('2024-07-24') ENGINE = InnoDB,
PARTITION p20240724 VALUES LESS THAN ('2024-07-25') ENGINE = InnoDB,
PARTITION p20240725 VALUES LESS THAN ('2024-07-26') ENGINE = InnoDB,
PARTITION p20240726 VALUES LESS THAN ('2024-07-27') ENGINE = InnoDB,
PARTITION p20240727 VALUES LESS THAN ('2024-07-28') ENGINE = InnoDB,
PARTITION p20240728 VALUES LESS THAN ('2024-07-29') ENGINE = InnoDB,
PARTITION p20240729 VALUES LESS THAN ('2024-07-30') ENGINE = InnoDB,
PARTITION p20240730 VALUES LESS THAN ('2024-07-31') ENGINE = InnoDB,
PARTITION p20240731 VALUES LESS THAN ('2024-08-01') ENGINE = InnoDB,
PARTITION p20240801 VALUES LESS THAN ('2024-08-02') ENGINE = InnoDB,
PARTITION p20240802 VALUES LESS THAN ('2024-08-03') ENGINE = InnoDB,
PARTITION p20240803 VALUES LESS THAN ('2024-08-04') ENGINE = InnoDB,
PARTITION p20240804 VALUES LESS THAN ('2024-08-05') ENGINE = InnoDB) */;

分区自动维护可见:

mysql分区自动维护(SpringBoot+MybatisPlus)

工具类

IP工具类

package com.zk.app.utils;

import javax.servlet.http.HttpServletRequest;
import java.net.InetAddress;
import java.net.UnknownHostException; /**
* @program: ZK
* @description: ip工具
* @author: zk
* @create: 2024-07-16 15:57
**/
public class IpUtil { private static final String UNKNOWN = "unknown";
private static final String LOCALHOST = "127.0.0.1";
private static final String SEPARATOR = ","; /**
* 获取当前请求ip地址
*
* @param request 请求对象
* @return 返回当前ip
*/
public static String getIpAddr(HttpServletRequest request) {
String ipAddress;
try {
ipAddress = request.getHeader("x-forwarded-for");
if (ipAddress == null || ipAddress.length() == 0 || UNKNOWN.equalsIgnoreCase(ipAddress)) {
ipAddress = request.getHeader("Proxy-Client-IP");
}
if (ipAddress == null || ipAddress.length() == 0 || UNKNOWN.equalsIgnoreCase(ipAddress)) {
ipAddress = request.getHeader("WL-Proxy-Client-IP");
}
if (ipAddress == null || ipAddress.length() == 0 || UNKNOWN.equalsIgnoreCase(ipAddress)) {
ipAddress = request.getRemoteAddr();
if (LOCALHOST.equals(ipAddress)) {
InetAddress inet = null;
try {
inet = InetAddress.getLocalHost();
} catch (UnknownHostException e) {
e.printStackTrace();
}
ipAddress = inet.getHostAddress();
}
}
// 对于通过多个代理的情况,第一个IP为客户端真实IP,多个IP按照','分割
// "***.***.***.***".length()
if (ipAddress != null && ipAddress.length() > 15) {
if (ipAddress.indexOf(SEPARATOR) > 0) {
ipAddress = ipAddress.substring(0, ipAddress.indexOf(","));
}
}
} catch (Exception e) {
ipAddress = "";
}
return ipAddress;
}
}

应用注解

在对应方法上加上注解即可

@OperationLog(operation = "获取列表", operationType = UserLogTypeEnum.SELECT)

不同类型可根据枚举类配置


结束

🎀java-自定义日志注解的更多相关文章

  1. 自定义日志注解 + AOP实现记录操作日志

      需求:系统中经常需要记录员工的操作日志和用户的活动日志,简单的做法在每个需要的方法中进行日志保存操作, 但这样对业务代码入侵性太大,下面就结合AOP和自定义日志注解实现更方便的日志记录   首先看 ...

  2. Java自定义日志输出文件

    Java自定义日志输出文件 日志的打印,在程序中是必不可少的,如果需要将不同的日志打印到不同的地方,则需要定义不同的Appender,然后定义每一个Appender的日志级别.打印形式和日志的输出路径 ...

  3. Spring自定义日志注解

    JDK1.5中引入注解,spring框架把java注解发扬光大 一  创建自定义注解 import java.lang.annotation.Retention; import java.lang.a ...

  4. java 自定义的注解有什么作用

    转自https://zhidao.baidu.com/question/1668622526729638507.html 自定义注解,可以应用到反射中,比如自己写个小框架. 如实现实体类某些属性不自动 ...

  5. Java 自定义日志写入

    /** * 将信息写入到日志 * @param content * @return * @throws IOException */ public static boolean writeLog(St ...

  6. springboot最新版本自定义日志注解和AOP

    LogAspectAnnotation @ControllerLogAspectAnnotation /** * * Define a log facet annotation * @author s ...

  7. java自定义注解类

    一.前言 今天阅读帆哥代码的时候,看到了之前没有见过的新东西, 比如java自定义注解类,如何获取注解,如何反射内部类,this$0是什么意思? 于是乎,学习并整理了一下. 二.代码示例 import ...

  8. java自定义注解实现前后台参数校验

    2016.07.26 qq:992591601,欢迎交流 首先介绍些基本概念: Annotations(also known as metadata)provide a formalized way ...

  9. Java自定义注解开发

    一.背景 最近在自己搞一个项目时,遇到可需要开发自定义注解的需求,对于没有怎么关注这些java新特性的来说,比较尴尬,索性就拿出一些时间,来进行研究下自定义注解开发的步骤以及使用方式.今天在这里记下, ...

  10. java自定义注解注解方法、类、属性等等【转】

    http://anole1982.iteye.com/blog/1450421 http://www.open-open.com/doc/view/51fe76de67214563b20b385320 ...

随机推荐

  1. Java中StringBuilder类常用的几个方法

    StringBuilder类 StringBuilder 类是 Java 中用于处理可变字符串的类,它提供了在字符串内部进行修改的方法,相比之下,String 类是不可变的,每次对字符串做修改都会创建 ...

  2. 理解ABP的领域驱动设计

    大家好,我是张飞洪,感谢您的阅读,我会不定期和你分享学习心得,希望我的文章能成为你成长路上的垫脚石,让我们一起精进. 关于玩转ABP框架相关的文章,之前在博客园陆续写了<ABP vNext系列文 ...

  3. 8款支持 C# 语言的 AI 辅助编程神器,高效编程利器!

    前言 在当今这个AI技术日新月异的时代,一股创新的浪潮正席卷着软件开发领域,其中AI辅助编程工具以其独特的魅力脱颖而出,成为了众多开发者不可或缺的得力助手.这些工具不仅能够显著提升开发效率,优化代码质 ...

  4. Scala高阶函数 2(以函数作为返回值,函数柯里化,应用函数)

    package com.wyh.day01 object ScalaFun4 { def main(args: Array[String]): Unit = { /** * 以函数作为返回值 */ d ...

  5. 机器学习 | 强化学习(6) | 策略梯度方法(Policy Gradient Method)

    6-策略梯度方法(Policy Gradient Method) 策略梯度概论(Introduction) 基于策略(Policy-Based) 的强化学习 对于上一节课(价值函数拟合)中采用参数\( ...

  6. AI回答:php中间件

    在PHP中,中间件(Middleware)是一种用于在处理请求和生成响应之间插入额外逻辑的机制.中间件通常用于执行诸如身份验证.日志记录.缓存.错误处理等任务.PHP本身并没有内置的中间件系统,但许多 ...

  7. 【代码设计】C# 实现 AOP 面向切面编程

    简单记录一下对AOP的认识,正文为3个部分 一.AOP由来 二.用DispatchProxy动态代理实现AOP 三.通过特性标记,处理多种不同执行前.执行后的逻辑编排 一.AOP 由来 IUserHe ...

  8. 【Matlab】求解复合材料层合板刚度矩阵及柔度矩阵

    1. matlab文件结构 2. main.m代码 clc clear; warning off; %% %铺层角度数组 angles=[0 90 0]; % ° %单层厚度 ply_thicknes ...

  9. Vulnhub-Hackme

    一.靶机搭建 选择扫描虚拟机 选择路径即可 二.信息收集 靶机信息 Name: hackme: 1 Date release: 18 Jul 2019 难度:初级,目标是通过web漏洞获得有限的权限访 ...

  10. Vue3状态管理终极指南:Pinia保姆级教程

    一.为什么选择Pinia?(Vuex对比分析) 1.1 核心优势解析 Composition API优先 :天然支持Vue3新特性,代码组织更灵活 TypeScript友好 :内置类型推导,无需额外类 ...