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

相关文章

C#中lock关键字的使用小结

《C#中lock关键字的使用小结》在C#中,lock关键字用于确保当一个线程位于给定实例的代码块中时,其他线程无法访问同一实例的该代码块,下面就来介绍一下lock关键字的使用... 目录使用方式工作原理注意事项示例代码为什么不能lock值类型在C#中,lock关键字用于确保当一个线程位于给定实例的代码块中时

MySQL 强制使用特定索引的操作

《MySQL强制使用特定索引的操作》MySQL可通过FORCEINDEX、USEINDEX等语法强制查询使用特定索引,但优化器可能不采纳,需结合EXPLAIN分析执行计划,避免性能下降,注意版本差异... 目录1. 使用FORCE INDEX语法2. 使用USE INDEX语法3. 使用IGNORE IND

C# $字符串插值的使用

《C#$字符串插值的使用》本文介绍了C#中的字符串插值功能,详细介绍了使用$符号的实现方式,文中通过示例代码介绍的非常详细,需要的朋友们下面随着小编来一起学习学习吧... 目录$ 字符使用方式创建内插字符串包含不同的数据类型控制内插表达式的格式控制内插表达式的对齐方式内插表达式中使用转义序列内插表达式中使用

flask库中sessions.py的使用小结

《flask库中sessions.py的使用小结》在Flask中Session是一种用于在不同请求之间存储用户数据的机制,Session默认是基于客户端Cookie的,但数据会经过加密签名,防止篡改,... 目录1. Flask Session 的基本使用(1) 启用 Session(2) 存储和读取 Se

springboot自定义注解RateLimiter限流注解技术文档详解

《springboot自定义注解RateLimiter限流注解技术文档详解》文章介绍了限流技术的概念、作用及实现方式,通过SpringAOP拦截方法、缓存存储计数器,结合注解、枚举、异常类等核心组件,... 目录什么是限流系统架构核心组件详解1. 限流注解 (@RateLimiter)2. 限流类型枚举 (

Java Thread中join方法使用举例详解

《JavaThread中join方法使用举例详解》JavaThread中join()方法主要是让调用改方法的thread完成run方法里面的东西后,在执行join()方法后面的代码,这篇文章主要介绍... 目录前言1.join()方法的定义和作用2.join()方法的三个重载版本3.join()方法的工作原

Spring AI使用tool Calling和MCP的示例详解

《SpringAI使用toolCalling和MCP的示例详解》SpringAI1.0.0.M6引入ToolCalling与MCP协议,提升AI与工具交互的扩展性与标准化,支持信息检索、行动执行等... 目录深入探索 Spring AI聊天接口示例Function CallingMCPSTDIOSSE结束语

Linux系统之lvcreate命令使用解读

《Linux系统之lvcreate命令使用解读》lvcreate是LVM中创建逻辑卷的核心命令,支持线性、条带化、RAID、镜像、快照、瘦池和缓存池等多种类型,实现灵活存储资源管理,需注意空间分配、R... 目录lvcreate命令详解一、命令概述二、语法格式三、核心功能四、选项详解五、使用示例1. 创建逻

在Java中使用OpenCV实践

《在Java中使用OpenCV实践》用户分享了在Java项目中集成OpenCV4.10.0的实践经验,涵盖库简介、Windows安装、依赖配置及灰度图测试,强调其在图像处理领域的多功能性,并计划后续探... 目录前言一 、OpenCV1.简介2.下载与安装3.目录说明二、在Java项目中使用三 、测试1.测

Python实现PDF按页分割的技术指南

《Python实现PDF按页分割的技术指南》PDF文件处理是日常工作中的常见需求,特别是当我们需要将大型PDF文档拆分为多个部分时,下面我们就来看看如何使用Python创建一个灵活的PDF分割工具吧... 目录需求分析技术方案工具选择安装依赖完整代码实现使用说明基本用法示例命令输出示例技术亮点实际应用场景扩