Spring boot项目实战之记录应用访问日志

1.说明

系统上线后往往我们需要知道都有哪些用户访问了应用的那些功能,以便更好的了解用户需求、防止恶意访问等。为此我们需要给应用添加记录访问日志的功能。下面就开始吧:

2.建表

CREATE TABLE `tb_operation_log` (
  `id` int(11) NOT NULL AUTO_INCREMENT,
  `oper_module` varchar(100) DEFAULT NULL COMMENT '操作模块',
  `oper_method` varchar(500) DEFAULT NULL COMMENT '操作方法',
  `oper_type` varchar(500) DEFAULT NULL COMMENT '操作类型',
  `oper_desc` varchar(1000) DEFAULT NULL COMMENT '操作描述',
  `oper_param` varchar(1000) DEFAULT NULL COMMENT '请求参数',
  `oper_ip` varchar(100) DEFAULT NULL COMMENT '请求参数',
  `oper_uri` varchar(1000) DEFAULT NULL COMMENT '请求uri',
  `create_time` datetime DEFAULT NULL COMMENT '操作时间',
  PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=74 DEFAULT CHARSET=utf8 COMMENT='操作日志表';

省略了model/dao/service代码,如有疑问可以交流。

3.创建注解

@Target(ElementType.METHOD) //注解放置的目标位置,METHOD是可注解在方法级别上
@Retention(RetentionPolicy.RUNTIME) //注解在哪个阶段执行
@Documented
public @interface OperLog {
    String operModule() default ""; // 操作模块
    String operType() default "";  // 操作类型
    String operDesc() default "";  // 操作说明
}

4.创建AOP(核心)

@Aspect
@Component
@Slf4j
public class OperLogAspect {
    @Autowired
    private OperationLogService operationLogService;

    /**
     * 操作日志切入点
     */
    @Pointcut("@annotation(com.sgytech.wecms.common.annotation.OperLog)")
    public void operLogPoinCut() {
    }

    /**
     * 拦截用户操作日志
     *
     * @param joinPoint 切入点
     * @param keys      返回结果
     */
    @AfterReturning(value = "operLogPoinCut()", returning = "keys")
    public void saveOperLog(JoinPoint joinPoint, Object keys) {

        // 获取RequestAttributes
        RequestAttributes requestAttributes = RequestContextHolder.getRequestAttributes();
        // 从获取RequestAttributes中获取HttpServletRequest的信息
        HttpServletRequest request = (HttpServletRequest) requestAttributes
                .resolveReference(RequestAttributes.REFERENCE_REQUEST);
        String requestIp = IpUtil.getRemoteIp(request);
        // 本机访问的不记录日志
        if(!"0:0:0:0:0:0:0:1".equals(requestIp)){
            OperationLog operlog = new OperationLog();
            try {
                // 从切面织入点处通过反射机制获取织入点处的方法
                MethodSignature signature = (MethodSignature) joinPoint.getSignature();
                // 获取切入点所在的方法
                Method method = signature.getMethod();
                // 获取操作
                OperLog opLog = method.getAnnotation(OperLog.class);
                if (opLog != null) {
                    String operModule = opLog.operModule();
                    String operType = opLog.operType();
                    String operDesc = opLog.operDesc();
                    operlog.setOperModule(operModule); // 操作模块
                    operlog.setOperType(operType); // 操作类型
                    operlog.setOperDesc(operDesc); // 操作描述
                }
                // 获取请求的类名
                String className = joinPoint.getTarget().getClass().getName();
                // 获取请求的方法名
                String methodName = method.getName();
                methodName = className + "." + methodName;

                operlog.setOperMethod(methodName); // 请求方法

                // 请求的参数
                Map<String, String> rtnMap = converMap(request.getParameterMap());
                // 将参数所在的数组转换成json
                String params = JSON.toJSONString(rtnMap);

                operlog.setOperParam(params); // 请求参数
                operlog.setOperIp(IpUtil.getRemoteIp(request)); // 请求IP
                operlog.setOperUri(request.getRequestURI()); // 请求URI
                log.info("当前的记录是:"+methodName+params);
                operationLogService.save(operlog);
            } catch (Exception e) {
                e.printStackTrace();
            }
        }

    }

    /**
     * 转换request 请求参数
     *
     * @param paramMap request获取的参数数组
     */
    public Map<String, String> converMap(Map<String, String[]> paramMap) {
        Map<String, String> rtnMap = new HashMap<String, String>();
        for (String key : paramMap.keySet()) {
            rtnMap.put(key, paramMap.get(key)[0]);
        }
        return rtnMap;
    }

5.controller中使用

以我这个博客项目中的文章详情接口为例,我要记录每次用户查看文章的详情行为日志,可以这样用:

@OperLog(operModule = "文章详情",operType = "查询",operDesc = "查询文章详情")
    @GetMapping("/{id}")
    public String detail(@PathVariable Integer id, Model model, HttpServletRequest request){
        // ...业务处理代码
    }

6.呈现记录的日志

直接上图吧:
在这里插入图片描述
日志的话就一般只读就行了 搜索可以做的详细点多条件检索、时间范围检索,支持删除就是了。

  • 3
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
Spring Boot是一个用于快速构建基于Java的应用程序的开源框架。它提供了许多开箱即用的功能和自动化配置,使得开发者可以更专注于业务逻辑的实现而不必花费太多时间和精力在基础设施方面。下面是一个Spring Boot项目实战的步骤: 1. 环境准备: - 安装Java Development Kit (JDK) - 下载并安装一个Java开发集成环境 (IDE),例如Eclipse或IntelliJ IDEA - 下载并安装Maven或Gradle构建工具 2. 创建Spring Boot项目: - 使用IDE创建一个新的Spring Boot项目 - 选择一个适合的项目模板,例如Web应用程序、RESTful服务或数据访问等 3. 配置项目: - 通过修改application.properties或application.yml文件来配置应用程序的属性,例如数据库连接信息、日志级别等 4. 定义实体类: - 创建实体类来映射数据库表格或数据对象 - 使用注解定义实体类的属性和关系 5. 创建数据访问层: - 创建一个Repository接口来定义对实体类的数据库操作方法 - 使用Spring Data JPA或其他ORM框架来实现Repository接口的具体实现 6. 创建业务逻辑层: - 创建一个Service类来实现业务逻辑 - 在Service类中调用Repository接口的方法来实现对数据库的操作 7. 创建控制器层: - 创建一个控制器类来处理HTTP请求和响应 - 使用注解定义控制器类的路由和请求处理方法 8. 运行和测试: - 启动应用程序并测试API的功能和性能 - 可以使用Postman或其他HTTP请求工具来进行API测试 以上是一个简单的Spring Boot项目实战的步骤。当然,具体的项目实战会根据实际需求的复杂程度和功能模块的多少而有所不同。希望这些步骤能够帮助你开始一个成功的Spring Boot项目

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值