张三进阶之路 | 基于Spring AOP的Log收集
前情提要 📌
张三对于公司的日志处理系统不满意,认为其性能不佳且功能有限。为了展示自己的能力和技术实力,他决定利用Spring AOP(面向切面编程)开发一个更高效的日志处理系统,并将其存储在Redis中。
首先,张三分析了现有日志处理系统的不足之处,如性能瓶颈、日志格式不统一、存储容量有限等。然后,他开始着手设计和实现一个新的日志处理系统。
📟 使用Spring AOP进行日志拦截:张三利用Spring AOP的切面功能,为需要记录日志的方法添加了一个切面。在这个切面中,他可以捕获方法的调用信息,如方法名、参数、返回值等,并将这些信息作为日志内容。
📟 日志格式化:为了确保日志的一致性和可读性,张三设计了一种统一的日志格式。他将日志分为不同的级别,如DEBUG、INFO、WARN和ERROR,并为每个级别设置了不同的颜色和标签。
📟 Redis存储:张三选择将日志存储在Redis中,因为Redis是一个高性能的键值存储系统,适合存储大量的日志数据。他为每个日志级别创建了一个Redis列表,用于存储相应级别的日志。同时,他还设置了一个定时任务,定期清理过期的日志数据,以保持存储空间的整洁。
📟 监控与告警:为了方便监控日志系统的运行状况,张三还开发了一个简单的监控界面,可以实时查看各个日志级别的数量、存储空间使用情况等信息。此外,他还设置了一些告警规则,当某个日志级别的数量超过阈值时,会自动发送告警通知给相关人员。
经过一段时间的努力,张三成功地完成了这个基于Spring AOP的日志处理系统,并将其部署到了生产环境。公司同事对他的工作表示赞赏,认为这个新的日志处理系统不仅提高了性能,还提供了更多有用的功能。这无疑突显了张三的技术能力和对公司的贡献。
以下是一个简化的代码实现示例,展示了如何使用Spring AOP和Redis来实现日志处理系统。
场景实现 📌
💵 创建一个日志切面类**LoggingAspect
**
在此过程中,我们创建了一个日志切面类LoggingAspect,它会拦截指定包路径下的所有方法调用。在方法调用完成后,它会将方法的调用信息(如方法名、参数、返回值等)作为日志内容,并将这些信息传递给LogService进行处理。
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.AfterReturning;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Pointcut;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;@Aspect
@Component
public class LoggingAspect {@Autowiredprivate LogService logService;@Pointcut("execution(* com.example.service.*.*(..))")public void logPointcut() {}@AfterReturning(pointcut = "logPointcut()", returning = "result")public void logAfterReturning(JoinPoint joinPoint, Object result) {logService.log(joinPoint, result);}
}
💵 创建一个日志服务类**LogService
****:**
LogService负责将日志内容存储到Redis中。在这个示例中,我们使用了RedisTemplate来操作Redis。我们将日志内容存储在名为log的Redis列表中。
import org.aspectj.lang.JoinPoint;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.stereotype.Service;@Service
public class LogService {@Autowiredprivate RedisTemplate<String, Object> redisTemplate;public void log(JoinPoint joinPoint, Object result) {String methodName = joinPoint.getSignature().getName();Object[] args = joinPoint.getArgs();String logMessage = String.format("Method: %s, Args: %s, Result: %s", methodName, Arrays.toString(args), result);// 将日志存储到RedisredisTemplate.opsForList().rightPush("log", logMessage);}
}
还可以为不同级别的日志创建不同的Redis列表:
public class LogService {// ...public void log(JoinPoint joinPoint, Object result, LogLevel logLevel) {String methodName = joinPoint.getSignature().getName();Object[] args = joinPoint.getArgs();String logMessage = String.format("Method: %s, Args: %s, Result: %s", methodName, Arrays.toString(args), result);// 根据日志级别将日志存储到不同的Redis列表中String redisKey = "log:" + logLevel.name().toLowerCase();redisTemplate.opsForList().rightPush(redisKey, logMessage);}
}
也可以修改日志格式化:
public class LogService {// ...public void log(JoinPoint joinPoint, Object result, LogLevel logLevel) {String methodName = joinPoint.getSignature().getName();Object[] args = joinPoint.getArgs();String logMessage = String.format("[%s] Method: %s, Args: %s, Result: %s", logLevel, methodName, Arrays.toString(args), result);// 根据日志级别将日志存储到不同的Redis列表中String redisKey = "log:" + logLevel.name().toLowerCase();redisTemplate.opsForList().rightPush(redisKey, logMessage);}
}
💵 配置RedisTemplate:
最后,我们配置了一个RedisTemplate Bean,用于序列化和反序列化Redis的key和value值。这样,我们就可以将日志内容以结构化的方式存储在Redis中,并在需要时方便地进行查询和分析。
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.data.redis.connection.RedisConnectionFactory;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.data.redis.serializer.GenericJackson2JsonRedisSerializer;
import org.springframework.data.redis.serializer.StringRedisSerializer;@Configuration
public class RedisConfig {@Beanpublic RedisTemplate<String, Object> redisTemplate(RedisConnectionFactory factory) {RedisTemplate<String, Object> template = new RedisTemplate<>();template.setConnectionFactory(factory);// 使用Jackson2JsonRedisSerializer来序列化和反序列化redis的value值GenericJackson2JsonRedisSerializer genericJackson2JsonRedisSerializer = new GenericJackson2JsonRedisSerializer();template.setValueSerializer(genericJackson2JsonRedisSerializer);template.setHashValueSerializer(genericJackson2JsonRedisSerializer);// 使用StringRedisSerializer来序列化和反序列化redis的key值StringRedisSerializer stringRedisSerializer = new StringRedisSerializer();template.setKeySerializer(stringRedisSerializer);template.setHashKeySerializer(stringRedisSerializer);template.afterPropertiesSet();return template;}
}
💵 定时任务:
我们创建了一个定时任务LogCleanupTask,它会定期清理过期的日志数据。我们使用了Spring的@Scheduled注解来实现定时任务,并使用RedisTemplate来操作Redis。在cleanupLogs方法中,我们遍历所有的日志列表,并根据日志的时间戳判断它们是否过期。如果过期,则将其从Redis中移除。
import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.stereotype.Component;@Component
public class LogCleanupTask {@Autowiredprivate RedisTemplate<String, Object> redisTemplate;@Scheduled(cron = "0 0 * * * ?") // 每小时执行一次public void cleanupLogs() {// 清理过期的日志数据,例如保留最近7天的日志String redisKeyPattern = "log:*";Set<String> keys = redisTemplate.keys(redisKeyPattern);for (String key : keys) {List<Object> logs = redisTemplate.opsForList().range(key, 0, -1);List<Object> logsToRemove = logs.stream().filter(log -> isExpired(log)).collect(Collectors.toList());redisTemplate.opsForList().remove(key, 0, logsToRemove);}}private boolean isExpired(Object log) {// 判断日志是否过期,例如根据日志的时间戳和当前时间进行比较// ...}
}
💵 监控界面:
我们创建了一个监控界面LogMonitorController,它可以实时查看日志数据。我们使用了Spring的@RestController注解来创建一个RESTful API,并使用RedisTemplate来操作Redis。在getLogs方法中,我们遍历所有的日志列表,并将它们以JSON格式返回给客户端。客户端可以使用这些数据来实时监控日志系统的运行状况。
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;@RestController
public class LogMonitorController {@Autowiredprivate RedisTemplate<String, Object> redisTemplate;@GetMapping("/monitor/logs")public Map<String, Object> getLogs() {Map<String, Object> logs = new HashMap<>();String redisKeyPattern = "log:*";Set<String> keys = redisTemplate.keys(redisKeyPattern);for (String key : keys) {List<Object> logList = redisTemplate.opsForList().range(key, 0, -1);logs.put(key, logList);}return logs;}
}
❗❗❗ 请注意!!! 这个示例仅用于演示如何使用Spring AOP和Redis实现日志处理系统。在实际项目中,需要根据具体需求进行更多的定制和优化。例如,可以为不同级别的日志创建不同的Redis列表,以便更好地管理和查询日志数据。此外,还可以考虑使用更高级的日志框架,如Logback或Log4j2,以实现更丰富的日志功能和更好的性能。
Get知识点 📌
📣 AOP概念:AOP(面向切面编程)是一种编程范式,它允许开发者在不修改原有代码的情况下,对程序的某些方面进行增强。AOP通过将横切关注点(如日志记录、事务管理、权限控制等)与业务逻辑分离,使得代码更加模块化和可维护。
📣 Spring AOP:Spring AOP是Spring框架中的一个重要组件,它提供了声明式的AOP支持。Spring AOP使用代理模式来实现AOP,可以通过JDK动态代理或CGLIB代理来创建代理对象。Spring AOP支持多种类型的切面,如前置通知、后置通知、异常通知、环绕通知等。
📣 @Aspect — 此注释将类定义为一个方面,即关注点的模块化。该方面包含建议和要点。
📣 @Joinpoint — 连接点是程序执行中可以应用方面的一个点。在 Spring AOP 中,连接点是方法调用。
📣 @Advice — 建议是某个方面在特定连接点上采取的行动。有几种类型的建议,例如“之前”、“之后”、“周围”等。
📣 @Pointcut — 切点是一组应应用方面的连接点。它定义了一个模式,该模式与方面应截获的方法相匹配。可以使用表达式或注释来定义切点。
下面是 Spring AOP 注解的示例:
@Aspect
@Component
public class LoggingAspect {@Before("execution(public * com.example.myapp.service.*.*(..))")public void logBefore(JoinPoint joinPoint) {System.out.println("Before " + joinPoint.getSignature().getName() + " method");}@After("execution(public * com.example.myapp.service.*.*(..))")public void logAfter(JoinPoint joinPoint) {System.out.println("After " + joinPoint.getSignature().getName() + " method");}@Around("execution(public * com.example.myapp.service.*.*(..))")public Object logAround(ProceedingJoinPoint joinPoint) throws Throwable {System.out.println("Before " + joinPoint.getSignature().getName() + " method");Object result = joinPoint.proceed();System.out.println("After " + joinPoint.getSignature().getName() + " method");return result;}@Pointcut("execution(public * com.example.myapp.service.*.*(..))")public void serviceMethods() {}
}
写在最后 📌
日志使用在现代软件开发中非常重要,它可以帮助开发者和系统管理员监控程序运行状态、排查问题和调试代码。但是,日志使用也存在一些缺点,如干扰员工工作、信息整理工作量大、主观色彩和日志格式不统一等。因此,在使用日志时,需要权衡其优缺点,选择合适的日志记录方法,并确保日志数据的准确性和完整性。
相关文章:
张三进阶之路 | 基于Spring AOP的Log收集
前情提要 📌 张三对于公司的日志处理系统不满意,认为其性能不佳且功能有限。为了展示自己的能力和技术实力,他决定利用Spring AOP(面向切面编程)开发一个更高效的日志处理系统,并将其存储在Redis中。 首先…...
ubuntu新装ubuntu,重启黑屏
现象:双系统电脑向移动硬盘安装Ubuntu系统后,重启黑屏并显示Minimal BASH-like line editing is supported. For the first word, TAB lists possible command completions. Anywhere else TAB lists possible device or file completions. 又拔下无法启…...
太极安全监控系统0.8
完善后的代码及功能详细介绍 完善后的代码 python import os import sys import subprocess import re import datetime import threading import tkinter as tk from tkinter import messagebox, simpledialog, ttk import scapy.all as scapy import whois import numpy as …...
E-清楚姐姐的布告规划【01背包】
就当一个01背包写就行,只不过需要保证不交叉,w[i]覆盖i点,用一个if来判断即可 #include<bits/stdc.h> #define int long long using namespace std; int w[5005]; int f[5005]; int t,n,m; signed main() {cin>>t;while(t--){…...
哪款宠物空气净化器噪音低?希喂、美的、安德迈测评分享
今年双11,宠物空气净化器到底应该如何选?在所有的家电品类里,宠物空气净化器算是比较特殊的那个,产品迭代太快,我们把今年双11在售的各大主流品牌的宠物空气净化器统一汇总整理,发现基本一多半都是24年下半…...
2024年10月23日第一部分
1.马小民要不要承担责任 2.主动 我就是那种平常沉默寡言孤僻内向自卑又宅又无趣,感觉不管在哪里都是比较边缘不合群的人。6月份遇到一个女生,还是人家主动加的我,断断续续聊了一个月就没下文了,可能我没谈过恋爱吧,快…...
医院信息化与智能化系统(9)
医院信息化与智能化系统(9) 这里只描述对应过程,和可能遇到的问题及解决办法以及对应的参考链接,并不会直接每一步详细配置 如果你想通过文字描述或代码画流程图,可以试试PlantUML,告诉GPT你的文件结构,让他给你对应的…...
逻辑回归与神经网络
从逻辑回归开始学习神经网络 神经网络直观上解释,就是由许多相互连接的圆圈组成的网络模型: 而逻辑回归可以看作是这个网络中的一个圆圈: 圆圈被称为神经元,整个网络被称为神经网络。 本节的任务是我们究竟如何理解具体的一个神…...
隨筆 20241024 Kafka 数据格式解析:批次头与数据体
Kafka作为分布式流处理平台,以其高吞吐量、可扩展性和强大的数据传输能力,成为了现代大数据和实时处理的核心组件之一。在Kafka中,数据的存储和传输遵循一种高效的结构化格式,主要由 批次头(Batch Header)和…...
【WiFi7】 支持wifi7的手机
数据来源 Smartphones with WiFi 7 - list of all latest phones 2024 Motorola Moto X50 Ultra 6.7" 1220x2712 Snapdragon 8s Gen 3 16GB RAM 1024 GB 4500 mAh a/b/g/n/ac/6e/7 Sony Xperia 1 VI 6.5" 1080x2340 Snapdragon 8 Gen 3 12GB RAM 512 G…...
LabVIEW偏振调制激光高精度测距系统
在航空航天、汽车制造、桥梁建筑等先进制造领域,许多大型零件的装配精度要求越来越高,传统的测距方法在面对大尺寸、高精度测量时,难以满足工业应用的要求。绝对测距技术在大尺度测量上往往会因受环境影响大、测距精度低而无法满足需求。基于…...
Python Pandas 数据分析的得力工具:简介
Python Pandas 数据分析的得力工具:简介 在如今的大数据与人工智能时代,数据的收集和处理能力变得至关重要。无论是在科学研究、商业分析还是人工智能领域,如何快速、高效地分析和处理数据都是不可忽视的课题。在众多的数据分析工具中&#…...
Llama 3.2-Vision 多模态大模型本地运行教程
Ollama 刚刚放出了对 Llama 3.2-Vision 的支持!这让人想起了新游戏发布带来的兴奋感——我期待着探索 Ollama 对 Llama 3.2-Vision 的支持。该模型不仅在自然语言理解方面表现出色,而且可以无缝处理图像,最好的部分是什么?它是免费…...
iOS 18.2 可让欧盟用户删除App Store、Safari、信息、相机和照片应用
升级到 iOS 18.2 之后,欧盟的 iPhone 用户可以完全删除一些核心应用程序,包括 App Store、Safari、信息、相机和 Photos 。苹果在 8 月份表示,计划对其在欧盟的数字市场法案合规性进行更多修改,其中一项更新包括欧盟用户删除系统应…...
照片怎么转换成pdf?盘点6种图片转pdf格式有效方法,直击要点!
照片怎么转换成pdf?在日常生活和工作中,我们难免会碰到需要将照片以pdf格式保存的情况,以便于更好的整理、分享或打印。虽然jpg格式的图片因其体积小而方便分享,但有时我们也希望将这些图片转换成pdf格式,以便于创建专…...
【Qt】Windows下Qt连接DM数据库
环境信息:W11 Qt5.12及以上 dm8 QODBC达梦 Windows环境创建ODBC数据源 使用 ODBC 方法访问 DM 数据库服务器之前,必须先配置 ODBC 数据源 在控制面板Windows工具中显示ODBC数据源管理器 ODBC数据源管理器标签 用户 DSN:添加、删除或配置本…...
2024 你还不会微前端吗 (上) — 从巨石应用到微应用
前言 微前端系列分为 上/下 两篇,本文为 上篇 主要还是了解微前端的由来、概念、作用等,以及基于已有的微前端框架进行实践,并了解微前端的核心功能所在,而在 下篇 中主要就是通过自定义实现一个微前端框架来加深理解。 微前端是…...
WPF+MVVM案例实战(三)- 动态数字卡片效果实现
1、创建项目 打开 VS2022 ,新建项目 Wpf_Examples,创建各层级文件夹,安装 CommunityToolkit.Mvvm 和 Microsoft.Extensions.DependencyInjectio NuGet包,完成MVVM框架搭建。搭建完成后项目层次如下图所示: 这里如何实现 MVVM 框…...
#网络安全#渗透测试# 渗透测试应用
网络安全渗透测试是一种重要的安全评估方法,用于发现和评估网络系统中的安全漏洞。在进行渗透测试时,需要注意以下几个关键点: 法律和道德考量 获得授权:在进行渗透测试之前,必须获得目标系统的正式授权。未经授权的测…...
MicroServer Gen8再玩 OCP万兆光口+IT直通之二
这个接上一篇,来个简单测试。 一、测试环境 PC端:Win10,网卡:万兆光纤(做都做了,都给接上),硬盘使用N年的三星SSD 840 交换机:磊科GS10,带两个万兆口 Gen…...
【JAVA面试题】Java和C++主要区别有哪些?各有哪些优缺点?
文章目录 强烈推荐前言区别:1. 语法和编程风格2.内存管理3.平台独立性4.性能5.指针和引用6.多线程7.使用场景 Java 的优缺点优点:缺点: C 的优缺点优点:缺点: 总结专栏集锦 强烈推荐 前些天发现了一个巨牛的人工智能学…...
保姆级教程!!教你通过【Pycharm远程】连接服务器运行项目代码
小罗碎碎念 这篇文章主要解决一个问题——我有服务器,但是不知道怎么拿来写代码,跑深度学习项目。确实,玩深度学习的成本比较高,无论是前期的学习成本,还是你需要具备的硬件成本,都是拦路虎。小罗没有办法…...
JMeter详细介绍和相关概念
JMeter是一款开源的、强大的、用于进行性能测试和功能测试的Java应用程序。 本篇承接上一篇 JMeter快速入门示例 , 对该篇中出现的相关概念进行详细介绍。 JMeter测试计划 测试计划名称和注释:整个测试脚本保存的名称,以及对该测试计划的注…...
如何使用Git
简介 一.git简介 Git是一个分布式版本控制工具,通常用来对软件开发过程中的源代码文件进行管理.通过Git仓库来存储和管理这些文件,Git仓库分两种: 本地仓库:开发人员自己电脑上的Git仓库远程仓库:远程服务器上的Git仓库 commit:提交,将本地文件和版本信息保存到本地仓库 p…...
Redis 哨兵 问题
前言 相关系列 《Redis & 目录》(持续更新)《Redis & 哨兵 & 源码》(学习过程/多有漏误/仅作参考/不再更新)《Redis & 哨兵 & 总结》(学习总结/最新最准/持续更新)《Redis & 哨兵…...
安卓基础001
前言 也是好久没有更新博客了,最近实习也是需要学习一些知识哈哈哈哈哈哈为了更好的发展嘛,咱们从客户端开始,过程可能有点像写前端,不喜勿喷,希望在学习的过程中也可以给大家带来一些简单得帮助吧....... tips:这里跳过安卓studio安装,大家可自行寻找教程 写的不详细,只是为了…...
shodan2:绕过shodan高级会员限制+metasploit批量验证漏洞
shodan2 shodanmetasploit批量验证漏洞 shodan的这个指令语法是特别多的,那么我不可能说一个个全部讲完,因为有的参数可能你一辈子都用不上,主要就是把一些红队最常用的参数给你讲完,今天我们看看怎么去查一个cve-2019-0708的一…...
【JAVA毕业设计】基于Vue和SpringBoot的母婴商城系统
本文项目编号 T 030 ,文末自助获取源码 \color{red}{T030,文末自助获取源码} T030,文末自助获取源码 目录 一、系统介绍二、演示录屏三、启动教程四、功能截图五、文案资料5.1 选题背景5.2 国内外研究现状5.3 可行性分析 六、核心代码6.1 查…...
探索Python安全字符串处理的奥秘:MarkupSafe库揭秘
文章目录 探索Python安全字符串处理的奥秘:MarkupSafe库揭秘第一部分:背景介绍第二部分:MarkupSafe是什么?第三部分:如何安装MarkupSafe?第四部分:MarkupSafe的简单使用方法1. 使用escape函数2.…...
Xcode真机运行正常,打包报错
1.问题: 老项目Xcode真机运行没问题,但但打包的时候却报了以下错误: some files could not be transferred (code 23) at /AppleInternal/Library/BuildRoots/4ff29661-3588-11ef-9513-e2437461156c/Library/Caches/com.apple.xbs/Sources/r…...
网站美工培训课程/网络优化公司哪家好
作者 | Raini 出品 | 图特摩斯(北京)科技有限公司 AbutionGraph是什么? AbutionGraph是一种能对历史和实时数据提供亚秒级别查询的多维图数据存储与分析平台。 AbutionGraph支持低延时的数据摄取,灵活的数据探索分析,高性能的…...
网站管理工作一般包括/成都网络营销
点击上方“Java基基”,选择“设为星标”做积极的人,而不是积极废人!每天 14:00 更新文章,每天掉亿点点头发...源码精品专栏 原创 | Java 2021 超神之路,很肝~中文详细注释的开源项目RPC 框架 Dubbo 源码解析网络应用框…...
wordpress自动发布模块/互联网营销师考证多少钱
分享一组Rpg Marker人物行走,游戏素材图片,共5张图片 上面的下载地址链接是图片,无法直接复制哦!下载请直接点击:游戏素材下载 或者复制以下链接:http://www.2gei.com/view/50.html...
绚丽的网站/网址导航大全
前言 今天我们来学习一下动态sql,看起来很NB的感觉。我们来看看官网是怎么来介绍动态sql的。动态 SQL 是 MyBatis 的强大特性之一。如果你使用过 JDBC 或其它类似的框架,你应该能理解根据不同条件拼接 SQL 语句有多痛苦,例如拼接时要确保不能…...
专门做网站的公司 南阳/google安卓手机下载
目前互联网上有无数个开源的建站程序可供大家选择使用,对现在的站长来说真的是容易多了,10年前我作网站的时候,一个小聊天程序也要自己一句一句的写,看看现在的开源程序,层出不穷。太多了也就不知道选哪个好了…...
怎么区分网站是模板做的/淘宝新店怎么快速做起来
为什么80%的码农都做不了架构师?>>> canvas 英音 /knvəs/ 美音 /knvəs/ 帆布, 画布 canvas的基本介绍 canvas是html5中新增的一个画布标签。这个标签的默认宽高为300*150设置canvas标签的宽高需要使用表格的形式,width和height…...