- VisualStudio2022插件的安装及使用-编程手把手系列文章
- pprof-在现网场景怎么用
- C#实现的下拉多选框,下拉多选树,多级节点
- 【学习笔记】基础数据结构:猫树
在平时使用到一些软件中,比如某宝或者某书,通过记录用户的行为来构建和分析用户的行为数据,同时也能更好优化产品设计和提升用户体验。比如在一个订单系统中,需要确定追踪用户的行为,比如:
上述行为就需要使用到日志系统来存储或者记录数据,Java 有几种日志方案,简单介绍几种实现方案,以及需要注意的点.
根据业务的不同,需要使用匹配到合适的日志方案。也需要注意几个问题:
Spring AOP 通过切面编程实现不修改原有代码,而动态添加功能的能力。这种方式有以下几个好处:
本文使用基于注解的 AOP,首先定义一个切面注解 AopTest
/**
* 切面注解
*/
@Target({ElementType.TYPE, ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
public @interface AopTest {
}
再创建通知 @Around :
@Around("@annotation(com.test.annotation.AopTest)")
public Object annotationTest(ProceedingJoinPoint joinPoint) throws Throwable {
log.info("执行前");
Object result = joinPoint.proceed(); // 执行目标方法
Object[] args = joinPoint.getArgs();
log.info("执行后");
return result;
}
一般都会在执行后添加日志即可,想要在那个方法或者接口加日志,只需要在方法上添加注解即可,比如在接口添加注解:
@GetMapping
@AopTest
public String first(String param) {
log.info("执行first方法");
return "result " + param;
}
请求接口后,就有如下的输出:
执行前
执行first方法
执行后
但是切面有一个问题,执行切面报错,方法也无法执行,就需要捕获异常,保证业务代码正常执行,改造一下上面的通知:
@Around("@annotation(com.test.annotation.AopTest)")
public Object annotationTest(ProceedingJoinPoint joinPoint) throws Throwable {log.info("执行前");
log.info("执行前");
Object result = joinPoint.proceed(); // 执行目标方法
try {
Object[] args = joinPoint.getArgs();
// 添加日志
log.info("执行后");
int a = 1/0;
} catch (Exception e) {
log.error(e.getMessage());
}
return result;
}
改造后,即使切面报错,也不会影响业务代码的执行了.
AOP 是同步执行的,如果日志添加是一个比较耗时的操作,也会影响接口的响应速度,此时可以使用异步的方式,比如消息队列.
总结一下,Spring AOP 有以下几个优点和缺点.
优点:
缺点及解决方案:
1、切面报错可能会影响业务代码 。
2、同步执行会影响接口响应速度 。
Spring 事件监听机制是一种发布-订阅模式,将事件的发布和监听解耦开来。通过这种机制,事件的发布者无需关注监听的逻辑,监听者也无需直接依赖发布者.
Spring 事件监听有三个部分组成:
自定义一个事件,继承 ApplicationEvent:
@Getter
@Setter
public class DemoEvent extends ApplicationEvent {
private String name;
public DemoEvent(Object source) {
super(source);
}
}
基于 @EventListener 注解监听事件:
@Component
@Slf4j
public class DemoEventListener {
@EventListener
public void handleEvent(DemoEvent event){
log.info(event.getName());
log.info("事件监听");
}
}
使用 applicationEventPublisher.publishEvent 方法发布事件, 。
@Autowired
private ApplicationEventPublisher applicationEventPublisher;
@Override
public void publish() {
// 执行业务代码
log.info("执行登录,当前时间 {}",LocalTime.now());
DemoEvent event = new DemoEvent(this);
event.setName("hello");
applicationEventPublisher.publishEvent(event);
log.info("完成登录,当前时间 {}",LocalTime.now());
log.info("执行 service");
}
配置好事件的发布和监听之后,在业务代码添加事件的发布,监听方法内添加日志的记录.
Spring 事件监听虽然解耦了发布和监听,只是解耦逻辑代码,两者还是同步执行,并且都是在同一个线程执行的,所以事件监听也无法解决添加日志报错,以及耗时的问题.
在事件监听添加延迟:
@EventListener
public void handleEvent(DemoEvent event){
try {
Thread.sleep(2000);
} catch (InterruptedException e) {
e.printStackTrace();
}
log.info(event.getName());
log.info("事件监听");
}
控制台输出:
执行登录,当前时间 16:52:30.799
hello
事件监听
完成登录,当前时间 16:52:33.799
执行 service
接口执行了 3 秒多,并且 publish 方法要等待监听方法执行完毕之后才能执行,说明事件监听是同步的.
在监听方法添加错误代码:
@EventListener
public void handleEvent(DemoEvent event){
int a = 1/0;
log.info(event.getName());
log.info("事件监听");
}
控制台输出:
执行登录,当前时间 17:10:08.396
java.lang.ArithmeticException: / by zero
监听方法报错,接口也报错,业务代码无法执行,说明监听方法报错会影响事件发布方法.
解决报错的问题,使用异常捕获即可。而延迟的问题,就需要使用到异步的操作,异步就是另启一个线程执行监听方法,异步除了能解决延迟的问题,也顺便解决了报错的问题.
实现异步在监听方法上添加 @Async 异步注解,监听方法添加延迟和错误代码:
执行登录,当前时间 17:21:50.971
完成登录,当前时间 17:21:50.974
执行 service
publish 方法既不会延迟,也不会因为监听报错影响执行,异步完美解决耗时和报错的问题 。
海量日志场景,消息队列是一个很好的选择,它也是解耦了发布者和订阅者,如果订阅者开启了手动确认,消费者也需要使用 try catch 捕获异常信息,确保消息能正常消费.
本文介绍几种日志记录的实现方案:
Spring AOP
:
事件监听 + 异步
:
消息队列
:
最后此篇关于Java日志记录几种实现方案的文章就讲到这里了,如果你想了解更多关于Java日志记录几种实现方案的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
我有一个网站。 必须登录才能看到里面的内容。 但是,我使用此代码登录。 doc = Jsoup.connect("http://46.137.207.181/Account/Login.aspx")
我正在尝试为我的域创建一个 SPF 记录并使我的邮件服务器能够对其进行评估。我在邮件服务器上使用 Postfix 并使用 policyd-spf (Python) 来评估记录。目前,我通过我的私有(p
我需要为负载平衡的 AWS 站点 mywebsite.com 添加 CName 记录。记录应该是: @ CNAME mywebsite.us-east-1.elb.amazon
我目前正在开发一个相当大的多层应用程序,该应用程序将部署在海外。虽然我希望它在解聚后不会折叠或爆炸,但我不能 100% 确定这一点。因此,如果我知道我可以请求日志文件,以准确找出问题所在以及原因,那就
我使用以下命令从我的网络摄像头录制音频和视频 gst-launch-0.10 v4l2src ! video/x-raw-yuv,width=640,height=480,framerate=30/1
我刚刚开始使用 ffmpeg 将视频分割成图像。我想知道是否可以将控制台输出信息保存到日志文件中。我试过“-v 10”参数,也试过“-loglevel”参数。我在另一个 SO 帖子上看到使用 ffmp
我想针对两个日期查询我的表并检索其中的记录。 我这样声明我的变量; DECLARE @StartDate datetime; DECLARE @EndDate datetime; 并像这样设置我的变量
在 javascript 中,我可以使用简单的 for 循环访问对象的每个属性,如下所示 var myObj = {x:1, y:2}; var i, sum=0; for(i in myObj) s
最近加入了一个需要处理大量代码的项目,我想开始记录和可视化调用图的一些流程,让我更好地理解一切是如何组合在一起的。这是我希望在我的理想工具中看到的: 每个节点都是一个函数/方法 如果一个函数可以调用另
如何使用反射在F#中创建记录类型?谢谢 最佳答案 您可以使用 FSharpValue.MakeRecord [MSDN]创建一个记录实例,但是我认为F#中没有任何定义记录类型的东西。但是,记录会编译为
关闭。这个问题不满足Stack Overflow guidelines .它目前不接受答案。 想改善这个问题吗?更新问题,使其成为 on-topic对于堆栈溢出。 3年前关闭。 Improve thi
我是 Sequelize 的新手并且遇到了一些语法问题。我制作了以下模型: // User sequelize.define('user', { name: { type: DataTyp
${student.name} Notify 这是我的output.jsp。请注意,我已经放置了一个链接“Notify”以将其转发到 display.jsp 上。但我不确定如何将 Stud
例如,这是我要做的查询: server:"xxx.xxx.com" AND request_url:"/xxx/xxx/xxx" AND http_X_Forwarded_Proto:(https O
我一直在开发大量 Java、PHP 和 Python。所有这些都提供了很棒的日志记录包(分别是 Log4J、Log 或logging)。这在调试应用程序时有很大帮助。特别是当应用程序 headless
在我的Grails应用程序中,我异步运行一些批处理过程,并希望该过程记录各种状态消息,以便管理员以后可以检查它们。 我考虑过将log4j JDBC附加程序用作最简单的解决方案,但是据我所知,它不使用D
我想将进入 MQ 队列的消息记录到数据库/文件或其他日志队列,并且我无法修改现有代码。是否有任何方法可以实现某种类似于 HTTP 嗅探器的消息记录实用程序?或者也许 MQ 有一些内置的功能来记录消息?
如果我有一条包含通用字段的记录,在更改通用字段时是否有任何方法可以模仿方便的 with 语法? 即如果我有 type User = // 'photo can be Bitmap or Url {
假设我有一个名为 Car 的自定义对象。其中的所有字段都是私有(private)的。 public class Car { private String mName; private
当记录具有特定字段时,我需要返回 true 的函数,反之亦然。示例: -record(robot, {name, type=industrial, ho
我是一名优秀的程序员,十分优秀!