SSM项目使用AOP技术进行日志记录

2024-09-08 12:52

本文主要是介绍SSM项目使用AOP技术进行日志记录,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

本步骤只记录完成切面所需的必要代码

本人开发中遇到的问题:

切面一直切不进去,最后发现需要在springMVC的核心配置文件中中开启注解驱动才可以,只在spring的核心配置文件中开启是不会在web项目中生效的。

之后按照下面的代码进行配置,然后前端在访问controller层中的路径时即可观察到日志已经被正常记录到数据库,代码中有部分注释,看不懂的可以参照注释。接下来进入正题

1、导入maven坐标
 <!--切面--><dependency><groupId>org.springframework</groupId><artifactId>spring-aop</artifactId><version>5.0.2.RELEASE</version></dependency><dependency><groupId>org.aspectj</groupId><artifactId>aspectjweaver</artifactId><version>1.8.7</version></dependency><!-- AspectJ Runtime --><dependency><groupId>org.aspectj</groupId><artifactId>aspectjrt</artifactId><version>1.8.7</version> </dependency>
2、核心文件配置,在springMVC.xml文件中开启AOP切面注解驱动
<?xml version="1.0" encoding="UTF-8"?>
<beansxmlns="http://www.springframework.org/schema/beans"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xmlns:context="http://www.springframework.org/schema/context"xmlns:mvc="http://www.springframework.org/schema/mvc"xmlns:aop="http://www.springframework.org/schema/aop"xsi:schemaLocation="http://www.springframework.org/schema/beanshttp://www.springframework.org/schema/beans/spring-beans.xsdhttp://www.springframework.org/schema/contexthttp://www.springframework.org/schema/context/spring-context.xsdhttp://www.springframework.org/schema/mvchttp://www.springframework.org/schema/mvc/spring-mvc.xsdhttp://www.springframework.org/schema/aophttp://www.springframework.org/schema/aop/spring-aop.xsd
"><!--    开启注解驱动--><context:annotation-config/><!--    开启组件扫描--><context:component-scan base-package="com.xszx"></context:component-scan><!--    开启切面注解驱动--><aop:aspectj-autoproxy></aop:aspectj-autoproxy>
</beans>
3、配置web.xml文件,指定spring容器的配置文件
<!DOCTYPE web-app PUBLIC"-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN""http://java.sun.com/dtd/web-app_2_3.dtd" ><web-app><display-name>Archetype Created Web Application</display-name><!--配置一个全局的参数:指定spring容器的配置文件 ServletContext--><context-param><param-name>contextConfigLocation</param-name><param-value>classpath:beans.xml</param-value></context-param><!--解决中文乱码的filter一定要放在最前面  --><filter><filter-name>CharacterEncodingFilter</filter-name><filter-class>org.springframework.web.filter.CharacterEncodingFilter</filter-class><!-- 配置encoding,告诉我们指定的编码格式 --><init-param><param-name>encoding</param-name><param-value>utf-8</param-value></init-param><init-param><!--是否强制设置request的编码为encoding,默认false,不建议更改--><param-name>forceRequestEncoding</param-name><param-value>false</param-value></init-param><init-param><!--是否强制设置response的编码为encoding,建议设置为true,下面有关于这个参数的解释--><param-name>forceResponseEncoding</param-name><param-value>false</param-value></init-param></filter><filter-mapping><filter-name>CharacterEncodingFilter</filter-name><url-pattern>/*</url-pattern></filter-mapping><!--配置监听器:创建spring容器对象--><listener><listener-class>org.springframework.web.context.ContextLoaderListener</listener-class></listener><!--定义一个request监听器--><listener><listener-class>org.springframework.web.context.request.RequestContextListener</listener-class></listener><servlet><servlet-name>SpringMVC</servlet-name><servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class><!--指定配置文件的路径--><init-param><param-name>contextConfigLocation</param-name><param-value>classpath:SpringMVC.xml</param-value></init-param></servlet><servlet-mapping><servlet-name>SpringMVC</servlet-name><url-pattern>/</url-pattern></servlet-mapping>
</web-app>
4、切面类
@Component
@Aspect
public class LogAop {@Resourceprivate LoginUserCountService loginUserCountService;@Autowiredprivate HttpServletRequest request;@Autowiredprivate ISysLogService sysLogService;private Date visitTime; //开始时间private Class clazz; //访问的类private Method method;//访问的方法//前置通知  主要是获取开始时间,执行的类是哪一个,执行的是哪一个方法@Before("execution(public * com.xszx.controller.*.*(..))")public void doBefore(JoinPoint jp) {try {visitTime = new Date(); // 当前时间就是开始访问的时间clazz = jp.getTarget().getClass(); // 具体要访问的类String methodName = jp.getSignature().getName(); // 获取访问的方法的名称Object[] args = jp.getArgs(); // 获取访问的方法的参数// 获取具体执行的方法的Method对象if (args == null || args.length == 0) {method = clazz.getMethod(methodName); // 只能获取无参数的方法} else {Class[] classArgs = new Class[args.length];for (int i = 0; i < args.length; i++) {classArgs[i] = args[i].getClass();}method = clazz.getMethod(methodName, classArgs);}} catch (NoSuchMethodException e) {//            e.printStackTrace();} catch (Exception e) {//            e.printStackTrace();}}//后置通知@After("execution(* com.xszx.controller.*.*(..))")public void doAfter(JoinPoint jp) {try {long time = new Date().getTime() - visitTime.getTime(); // 获取访问的时长String url = "";// 获取urlif (clazz != null && method != null && clazz != LogAop.class) {RequestMapping methodAnnotation = method.getAnnotation(RequestMapping.class);if (methodAnnotation != null) {String[] methodValue = methodAnnotation.value();url = methodValue[0];// 获取访问的ipString ip = request.getRemoteAddr();// 登录时将用户信息存入session,现在从session中获取当前操作的用户HttpSession httpSession = request.getSession();User user = (User)httpSession.getAttribute("loginUser");String userEmail = "";if (user == null){userEmail = "无登录用户";}else {userEmail = user.getEmail();}// 将日志相关信息封装到SysLog对象SysLog sysLog = new SysLog();sysLog.setExecutionTime(time); // 执行时长sysLog.setIp(ip);sysLog.setMethod("[类名] " + clazz.getName() + "[方法名] " + method.getName());sysLog.setUrl(url);sysLog.setUsername(userEmail);sysLog.setVisitTime(visitTime);// 调用Service完成操作sysLogService.save(sysLog);}}} catch (Exception e) {//            e.printStackTrace();}}
}
5、beans层
public class SysLog {private Integer id;private Date visitTime;private String visitTimeStr;private String username;private String ip;private String url;private Long executionTime;private String method;//记得写getter,setter方法,构造方法,toString方法
}
6、dao层
public interface ISysLogMapper {void save(SysLog sysLog) throws Exception;List<SysLog> findAll() throws Exception;
}
7、service层
@Service
public class SysLogServiceImpl implements ISysLogService {private SqlSessionTemplate sqlSessionTemplate;@Autowiredpublic void setSqlSessionTemplate(SqlSessionTemplate sqlSessionTemplate) {this.sqlSessionTemplate = sqlSessionTemplate;}@Overridepublic List<SysLog> findAll() throws Exception {ISysLogMapper iSysLogMapper = sqlSessionTemplate.getMapper(ISysLogMapper.class);return iSysLogMapper.findAll();}@Overridepublic void save(SysLog sysLog) throws Exception {ISysLogMapper iSysLogMapper = sqlSessionTemplate.getMapper(ISysLogMapper.class);iSysLogMapper.save(sysLog);}
}
8、mapper层
<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapperPUBLIC "-//mybatis.org//DTD Mapper 3.0//EN""http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.xszx.dao.ISysLogMapper"><insert id="save">insert into syslog(visitTime,username,ip,url,executionTime,method)values(#{visitTime},#{username},#{ip},#{url},#{executionTime},#{method})</insert><select id="findAll" resultType="com.xszx.beans.SysLog">select * from sysLog</select>
</mapper>
9、数据库表结构

执行完后进行测试即可,当前扫描的路径是controller下的所有方法,所以随便进入一个即可打印出对于的日志记录。  

这篇关于SSM项目使用AOP技术进行日志记录的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1148164

相关文章

Java中流式并行操作parallelStream的原理和使用方法

《Java中流式并行操作parallelStream的原理和使用方法》本文详细介绍了Java中的并行流(parallelStream)的原理、正确使用方法以及在实际业务中的应用案例,并指出在使用并行流... 目录Java中流式并行操作parallelStream0. 问题的产生1. 什么是parallelS

Linux join命令的使用及说明

《Linuxjoin命令的使用及说明》`join`命令用于在Linux中按字段将两个文件进行连接,类似于SQL的JOIN,它需要两个文件按用于匹配的字段排序,并且第一个文件的换行符必须是LF,`jo... 目录一. 基本语法二. 数据准备三. 指定文件的连接key四.-a输出指定文件的所有行五.-o指定输出

Linux jq命令的使用解读

《Linuxjq命令的使用解读》jq是一个强大的命令行工具,用于处理JSON数据,它可以用来查看、过滤、修改、格式化JSON数据,通过使用各种选项和过滤器,可以实现复杂的JSON处理任务... 目录一. 简介二. 选项2.1.2.2-c2.3-r2.4-R三. 字段提取3.1 普通字段3.2 数组字段四.

Linux kill正在执行的后台任务 kill进程组使用详解

《Linuxkill正在执行的后台任务kill进程组使用详解》文章介绍了两个脚本的功能和区别,以及执行这些脚本时遇到的进程管理问题,通过查看进程树、使用`kill`命令和`lsof`命令,分析了子... 目录零. 用到的命令一. 待执行的脚本二. 执行含子进程的脚本,并kill2.1 进程查看2.2 遇到的

Java AOP面向切面编程的概念和实现方式

《JavaAOP面向切面编程的概念和实现方式》AOP是面向切面编程,通过动态代理将横切关注点(如日志、事务)与核心业务逻辑分离,提升代码复用性和可维护性,本文给大家介绍JavaAOP面向切面编程的概... 目录一、AOP 是什么?二、AOP 的核心概念与实现方式核心概念实现方式三、Spring AOP 的关

详解SpringBoot+Ehcache使用示例

《详解SpringBoot+Ehcache使用示例》本文介绍了SpringBoot中配置Ehcache、自定义get/set方式,并实际使用缓存的过程,文中通过示例代码介绍的非常详细,对大家的学习或者... 目录摘要概念内存与磁盘持久化存储:配置灵活性:编码示例引入依赖:配置ehcache.XML文件:配置

Java 虚拟线程的创建与使用深度解析

《Java虚拟线程的创建与使用深度解析》虚拟线程是Java19中以预览特性形式引入,Java21起正式发布的轻量级线程,本文给大家介绍Java虚拟线程的创建与使用,感兴趣的朋友一起看看吧... 目录一、虚拟线程简介1.1 什么是虚拟线程?1.2 为什么需要虚拟线程?二、虚拟线程与平台线程对比代码对比示例:三

k8s按需创建PV和使用PVC详解

《k8s按需创建PV和使用PVC详解》Kubernetes中,PV和PVC用于管理持久存储,StorageClass实现动态PV分配,PVC声明存储需求并绑定PV,通过kubectl验证状态,注意回收... 目录1.按需创建 PV(使用 StorageClass)创建 StorageClass2.创建 PV

vite搭建vue3项目的搭建步骤

《vite搭建vue3项目的搭建步骤》本文主要介绍了vite搭建vue3项目的搭建步骤,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学... 目录1.确保Nodejs环境2.使用vite-cli工具3.进入项目安装依赖1.确保Nodejs环境

Redis 基本数据类型和使用详解

《Redis基本数据类型和使用详解》String是Redis最基本的数据类型,一个键对应一个值,它的功能十分强大,可以存储字符串、整数、浮点数等多种数据格式,本文给大家介绍Redis基本数据类型和... 目录一、Redis 入门介绍二、Redis 的五大基本数据类型2.1 String 类型2.2 Hash