实现用户操作日志记录

这篇具有很好参考价值的文章主要介绍了实现用户操作日志记录。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

Java记录操作日志

java自带的日志框架是java.util.logging(JUL),从JDK1.4(2002)开始捆绑在JDK中。可以使用JUL来记录操作日志。以下是使用JUL记录事务的示例:

// java.util.logging
java.util.logging.Logger logger = java.util.logging.Logger.getLogger (this.getClass ().getName ());
logger.info ("This is an info message");
logger.severe ("This is an error message"); // == ERROR

操作日志和系统日志的区别

系统日志:统日志主要是为开发排查问题提供依据,一般打印在日志文件中;系统日志的可读性要求没那么高,日志中会包含代码的信息,比如在某个类的某一行打印了一个日志。
操作日志:主要是对某个对象进行新增操作或者修改操作后记录下这个新增或者修改,操作日志要求可读性比较强,因为它主要是给用户看的,比如订单的物流信息,用户需要知道在什么时间发生了什么事情。

实现操作日志共能一般有两种方法

  1. 第一种就是很传统的做法,就是在每个模块进行插入日志的操作,这种方法虽然实现巨鹿用户操作,但是很繁琐,基本是重复的工作。
  2. 第二种就是使用spring的AOP来实现记录用户操作,也是现在最流行的写法。它的优势在于这种记录用户操作的代码独立于其他业务逻辑代码,不仅实现了解耦,而且避免了冗余代码。

操作日志表

这里我就简单记录一下基本的信息。
操作日志怎么实现,代码,java,servlet,spring,aop

需要的一些工具类

获取Ip地址

package com.kl.util;

import org.springframework.web.context.request.RequestAttributes;
import org.springframework.web.context.request.RequestContextHolder;
import org.springframework.web.context.request.ServletRequestAttributes;

import javax.servlet.http.HttpServletRequest;
import java.net.InetAddress;
import java.net.UnknownHostException;

/**
 * @author Wen先森
 * @version 1.0
 * @date 2023/2/28 10:25
 */
public class IpUtils {

    /**
     * 获取Ip地址
     * @return
     */
    public static String getIpAddress() {
        HttpServletRequest request = null;

        try {
            RequestAttributes requestAttributes = RequestContextHolder.currentRequestAttributes();
            request = ((ServletRequestAttributes)requestAttributes).getRequest();
        } catch (Exception var2) {
            return null;
        }

        return getIpAddr(request);
    }

    /**
     * 获取Ip地址
     * @param request
     * @return
     */
    public static String getIpAddr(HttpServletRequest request)
    {
        if (request == null)
        {
            return "unknown";
        }
        String ip = request.getHeader("x-forwarded-for");
        if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip))
        {
            ip = request.getHeader("Proxy-Client-IP");
        }
        if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip))
        {
            ip = request.getHeader("X-Forwarded-For");
        }
        if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip))
        {
            ip = request.getHeader("WL-Proxy-Client-IP");
        }
        if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip))
        {
            ip = request.getHeader("X-Real-IP");
        }

        if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip))
        {
            ip = request.getRemoteAddr();
        }

        return "0:0:0:0:0:0:0:1".equals(ip) ? "127.0.0.1" : ip;
    }

获取浏览器和操作系统

		<!-- 解析客户端操作系统、浏览器等 -->
		<dependency>
			<groupId>eu.bitwalker</groupId>
			<artifactId>UserAgentUtils</artifactId>
			<version>1.21</version>
		</dependency>
        UserAgent userAgent = UserAgent.parseUserAgentString(ServletUtils.getRequest().getHeader("User-Agent"));
        // 获取客户端操作系统
        String os = userAgent.getOperatingSystem().getName();
        // 获取客户端浏览器
        String browser = userAgent.getBrowser().getName();

第一种:传统方法

  1. 新增日志实体类、dao层 接口
  2. 这里我是将添加日志编写到工具类中
package com.kl.util;

import com.kl.entity.IotMSUser;
import com.kl.entity.OprLog;
import com.kl.repo.OprLogRepo;
import eu.bitwalker.useragentutils.UserAgent;
import org.springframework.util.ObjectUtils;

/**
 * @author Wen先森
 * @version 1.0
 * @date 2023/2/27 19:48
 */
public class LogUtils {
    /**
     * 添加操作日志
     * @param content
     */
    public static void insertLog (String content) {
        UserAgent userAgent = UserAgent.parseUserAgentString(ServletUtils.getRequest().getHeader("User-Agent"));
        // 获取客户端操作系统
        String os = userAgent.getOperatingSystem().getName();
        // 获取客户端浏览器
        String browser = userAgent.getBrowser().getName();
        // 获取用户信息,可以从自己的项目中token中去取,这里就不再描写
        Integer userId=1;
        // 获取Ip地址
        String ipAddress = IpUtils.getIpAddress();
        OprLog oprLog = new OprLog();
        oprLog.setIp(ipAddress);
        oprLog.setUserAgent(browser);
        oprLog.setUserId(userId);
        oprLog.setContent(content);
        oprLog.setCreateTime(FormatUtil.formatDate());
        // 这里我才用的是JPA,大家也可以换成自己熟悉的框架去添加
        OprLogRepo oprLogRepo = SpringUtils.getBean(OprLogRepo.class);
        oprLogRepo.save(oprLog);
    }
}
  1. 在controller层的某一个方法调用该工具类记录操作日志。
    操作日志怎么实现,代码,java,servlet,spring,aop

第二种:AOP实现记录用户操作

  1. 在pom.xml中添加AOP依赖
		<!-- aop -->
		<dependency>
			<groupId>org.springframework.boot</groupId>
			<artifactId>spring-boot-starter-aop</artifactId>
		</dependency>
  1. 自定义操作日志注解
package com.kl.interceptor.aspectj;

import java.lang.annotation.*;

/**
 * @author Wen先森
 * @version 1.0
 * @date 2023/3/6 16:03
 * @description: 自定义操作日志注解
 */
@Target(ElementType.METHOD)//注解放置的目标位置即方法级别
@Retention(RetentionPolicy.RUNTIME)//注解在哪个阶段执行
@Documented
public @interface OprLog {
    /**
     * 操作内容
     */
    public String content() default "";
}

  1. 自定义操作日志切面类,该类是将操作日志保存到数据库
package com.kl.interceptor.aspectj;

import com.kl.entity.IotMSUser;
import com.kl.repo.OprLogRepo;
import com.kl.util.*;
import eu.bitwalker.useragentutils.UserAgent;
import lombok.extern.slf4j.Slf4j;
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.Signature;
import org.aspectj.lang.annotation.AfterReturning;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Pointcut;
import org.aspectj.lang.reflect.MethodSignature;
import org.springframework.stereotype.Component;
import org.springframework.util.ObjectUtils;
import org.springframework.web.context.request.RequestAttributes;
import org.springframework.web.context.request.RequestContextHolder;

import javax.servlet.http.HttpServletRequest;
import java.lang.reflect.Method;
import java.sql.Timestamp;

/**
 * @author Wen先森
 * @date 2023/3/6 16:18
 * @version 1.0
 * @description: 操作日志切面处理类
 */
@Slf4j
@Aspect
@Component
public class OperationLogAspect {

    /**
     * 设置操作日志切入点   在注解的位置切入代码
     */
    @Pointcut("@annotation(com.kl.interceptor.aspectj.OprLog)")
    public void oprLogPointCut() {
    }


    /**
     * 记录操作日志
     * @param joinPoint 方法的执行点
     * @param result  方法返回值
     * @throws Throwable
     */
    
    @AfterReturning(returning  = "result", value = "oprLogPointCut()")
    public void saveOperLog(JoinPoint joinPoint, Object result){
        try{
            // 获得注解
            OprLog controllerLog = getAnnotationLog(joinPoint);
            if (controllerLog == null)
            {
                return;
            }
            //将返回值转换成map集合
            Map<String, String> map = (Map<String, String>) result;
            // 返回值信息(根据需求决定是否记录)
            String msg = MapUtils.getString(map, "msg");
            UserAgent userAgent = UserAgent.parseUserAgentString(ServletUtils.getRequest().getHeader("User-Agent"));
            // 获取客户端操作系统
            String os = userAgent.getOperatingSystem().getName();
            // 获取客户端浏览器
            String browser = userAgent.getBrowser().getName();
            // 获取用户信息
            IotMSUser user = TokenUtil.getUserId();
            // 用户ID,"-1" 代表没有登陆系统用户
            Integer userId=-1;
            if (!ObjectUtils.isEmpty(user)){
                userId=user.getId();
            }
            // 获取Ip地址
            String ipAddress = IpUtils.getIpAddress();
            com.kl.entity.OprLog oprLog = new com.kl.entity.OprLog();
            oprLog.setIp(ipAddress);
            oprLog.setUserAgent(browser);
            oprLog.setUserId(userId);
            oprLog.setContent(controllerLog.content());
            oprLog.setCreateTime(FormatUtil.formatDate());
            OprLogRepo oprLogRepo = SpringUtils.getBean(OprLogRepo.class);
            oprLogRepo.save(oprLog);
        }catch (Exception e){
            // 记录本地异常日志
            log.error("==前置通知异常==");
            log.error("异常信息:{}", e.getMessage());
        }
    }

    /**
     * 是否存在注解,如果存在就获取
     */
    private OprLog getAnnotationLog(JoinPoint joinPoint) throws Exception
    {
        Signature signature = joinPoint.getSignature();
        MethodSignature methodSignature = (MethodSignature) signature;
        Method method = methodSignature.getMethod();

        if (method != null)
        {
            return method.getAnnotation(OprLog.class);
        }
        return null;
    }

}
  1. 在controller层的某一个方法加入@OprLog注解
    操作日志怎么实现,代码,java,servlet,spring,aop
最终结果

操作日志怎么实现,代码,java,servlet,spring,aop文章来源地址https://www.toymoban.com/news/detail-618536.html

到了这里,关于实现用户操作日志记录的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 用户登录后IP记录日志的六种实现方案探讨

    之前大群里有小伙伴在讨论用户IP日志记录的一些方案,也有小伙伴在做这个需求,私底下跟我咨询过,所以在此特地汇总梳理一下。 ### 方案1 在登录业务中直接记录用户每次登录的IP日志,如下图所示: 用户请求登录的Controller,原先用户直接调用登录的service,这里假设用

    2024年01月17日
    浏览(43)
  • 学会使用Java Web实现:用户登录功能(JSP+Servlet方式)

    登录页面 login.jsp ,输入用户名和密码后,跳转到登录处理程序 LoginServlet 进行业务逻辑处理,登录成功,跳转到登录成功页面 success.jsp ,否则跳转到登录失败页面 failure.jsp 。 创建 Java Enterprise 项目,添加 Web Application 功能 设置项目名与保存位置 单击【Finish】按钮 在项目结

    2024年02月11日
    浏览(64)
  • 学会使用Java Web实现:用户登录(JSP+Servlet+DB)方式

    总体上采用MVC架构。登录页面 login.jsp ,输入用户名和密码后,跳转到登录处理程序 LoginServlet 进行业务逻辑处理,调用服务层,服务层调用数据访问层(DAO),连接数据库,查询数据库,以此判断是否登录成功。登录成功,跳转到登录成功页面 success.jsp ,否则跳转到登录失

    2024年02月02日
    浏览(48)
  • SpringBoot项目如何优雅的实现操作日志记录

    在实际开发当中,对于某些关键业务,我们通常需要记录该操作的内容,一个操作调一次记录方法,每次还得去收集参数等等,会造成大量代码重复。 我们希望代码中只有业务相关的操作,在项目中使用注解来完成此项功能。 通常就是使用Spring中的AOP特性来实现的,那么在

    2024年01月18日
    浏览(45)
  • 通过Annotation将用户操作记录到数据库表功能实现

    一、背景         在用户对我们所开发的系统访问的时候,需要我们的系统具有强大的健壮性,使得给与用户的体验感十足。在业务开发的过程中,我们通过将几个相关的操作绑定成一个事件,使得安全性以及数据的前后一致性得到提高。但是在溯源方面,我们往往没有很好

    2024年03月13日
    浏览(89)
  • spring boot 使用AOP+自定义注解+反射实现操作日志记录修改前数据和修改后对比数据,并保存至日志表

    使用FieldMeta自定义注解,看个人业务自行觉得是否需要重新定义实体 实现类 :通过该实现类获取更新前后的数据。 该实现类的实现原理为:获取入参出入的id值,获取sqlSessionFactory,通过sqlSessionFactory获取selectByPrimaryKey()该方法,执行该方法可获取id对应数据更新操作前后的数

    2024年01月23日
    浏览(53)
  • 用户信息、账号设置。换做是你会怎么做代码实现?

    在网站的应用场景下,对于用户的信息展示与账号设置的都是一个绕不过的问题。解决这个问题,最重要的是,怎么拦截未登录用户进入这个功能内?这就意味着必须在每次动态请求时,校验登录状态以及让请求持有用户(在服务器端可查询到对应用户)。当然,我们必须考

    2023年04月13日
    浏览(39)
  • 利用servlet+Javabean+JSP 实现用户登陆案例

    ​ 完成用户登录功能,登录界面必须要有 账号、密码、验证码。 ​ 后端技术要求为 Servlet.登录验证通过数据库进行验证数据库操作使用原生JDBC,前端 htmljsp、css、js 都可以,不能使用框架。代码需满足 Java 编程规范,注释需要清晰明了。 登陆首页 验证码输入错误 登陆失败

    2024年02月04日
    浏览(46)
  • 使用SpringBoot AOP记录操作日志和异常日志

    平时我们在做项目时经常需要对一些重要功能操作记录日志,方便以后跟踪是谁在操作此功能;我们在操作某些功 能时也有可能会发生异常,但是每次发生异常要定位原因我们都要到服务器去查询日志才能找到,而且也不能对发 生的异常进行统计,从而改进我们的项目,要

    2024年02月04日
    浏览(44)
  • AOP案例-记录日志操作

    目录 案例 操作日志 思路分析 步骤 准备 编码 将案例中的增删改接口的操作日志记录到数据表中 操作日志 日志信息包括:操作人、操作时间、执行方法的全类名、执行方法名、方法运行时的参数、返回值、犯法运行时长。 思路分析 需要对于所有业务类中的增删改查方法统

    2024年02月09日
    浏览(38)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包