-
Notifications
You must be signed in to change notification settings - Fork 1.7k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Memory Usage Increase in Log4j2 2.24.3 with Async Logging Due to RingBufferLogEvent #3569
Comments
Hi @shaonan666,
You should check the following options:
Note Log4j uses a fuzzy-matching algorithm to match both the official spelling of the configuration properties and some obsolete spellings. You should look if your configuration properties have names similar to those above, that start with |
Hi @ppkarwasz My program isn’t a web app, so Constants#isWebApp returns false at runtime, setting Additionally, I’ve noticed some version differences:
Based on these observations, I have the some questions:
|
Up to version
In version Since both implementations use immutable objects logging-log4j2/log4j-core/src/main/java/org/apache/logging/log4j/core/async/RingBufferLogEvent.java Lines 451 to 459 in 93afabc
Your log events retain instances of
The use of Right now, my best bet on why your system behaves the way he does is that |
Hi @ppkarwasz I checked the dump file and code, and it seems
In version 2.24.3, using ProviderUtil.getProvider().getThreadContextMapInstance(), I got DefaultThreadContextMap because GC_FREE_THREAD_CONTEXT_PROPERTY isn’t enabled logging-log4j2/log4j-core/src/main/java/org/apache/logging/log4j/core/impl/Log4jProvider.java Lines 113 to 116 in 4aa5efe
logging-log4j2/log4j-core/src/main/java/org/apache/logging/log4j/core/impl/Log4jProvider.java Lines 127 to 129 in 4aa5efe
![]()
In version 2.23.1, using ThreadContext.getThreadContextMap() (which returns null for the default map), I got
logging-log4j2/log4j-api/src/main/java/org/apache/logging/log4j/spi/ThreadContextMapFactory.java Lines 139 to 146 in fea2a71
Additionally, I couldn’t find |
I was looking at the heap dump you shared before: I am not able to reproduce this state, unless I add an Additional Context Property to my logger configuration: <Root level="INFO">
<AppenderRef ref="FILE"/>
<Property name="key" value="value"/>
</Root> Can you share your |
Hi @ppkarwasz <?xml version="1.0" encoding="UTF-8"?>
<Configuration status="INFO">
<properties>
<Property name="log4j2.path" value="${sys:logging.path:[email protected]@}"/>
<Property name="log4j2.immediateFlush" value="${sys:logging.immediateFlush:[email protected]@}"/>
<Property name="log4j2.level" value="${sys:logging.level:[email protected]@}"/>
<Property name="log4j2.max.file.size" value="xxxMB"/>
<Property name="log4j2.max.file.count" value="xxx"/>
<Property name="log4j2.max.file.time" value="xxx"/>
<Property name="PID">????</Property>
<Property name="LOG_EXCEPTION_CONVERSION_WORD">%xwEx</Property>
<Property name="LOG_LEVEL_PATTERN">%5p</Property>
<Property name="LOG_DATEFORMAT_PATTERN">yyyy-MM-dd HH:mm:ss.SSS</Property>
<Property name="CONSOLE_LOG_PATTERN">xxxx</Property>
<Property name="FILE_LOG_PATTERN">xxx</Property>
</properties>
<Appenders>
<Console name="Console" target="SYSTEM_OUT">
<PatternLayout pattern="${CONSOLE_LOG_PATTERN}"/>
</Console>
<RollingRandomAccessFile name="FILE_INFO"
fileName="${log4j2.path}/info.log"
filePattern="${log4j2.path}/archive/history_info-%d{yyyy-MM-dd}.%i.zip"
immediateFlush="${log4j2.immediateFlush}">
<PatternLayout pattern="${FILE_LOG_PATTERN}"/>
<Policies>
<TimeBasedTriggeringPolicy interval="1" modulate="true"/>
<SizeBasedTriggeringPolicy size="${log4j2.max.file.size}"/>
</Policies>
<DefaultRolloverStrategy max="${log4j2.max.file.count}">
<Delete basePath="${log4j2.path}/archive/" maxDepth="2">
<IfFileName glob="*.zip"/>
<IfLastModified age="${log4j2.max.file.time}"/>
</Delete>
</DefaultRolloverStrategy>
</RollingRandomAccessFile>
<RollingRandomAccessFile name="FILE_ERROR"
fileName="${log4j2.path}/error.log"
filePattern="${log4j2.path}/archive/history_error-%d{yyyy-MM-dd}.%i.zip"
immediateFlush="${log4j2.immediateFlush}">
<Filters>
<ThresholdFilter level="WARN" onMatch="ACCEPT" onMismatch="DENY"/>
</Filters>
<PatternLayout pattern="${FILE_LOG_PATTERN}"/>
<Policies>
<TimeBasedTriggeringPolicy interval="1" modulate="true"/>
<SizeBasedTriggeringPolicy size="${log4j2.max.file.size}"/>
</Policies>
<DefaultRolloverStrategy max="${log4j2.max.file.count}">
<Delete basePath="${log4j2.path}/archive/" maxDepth="2">
<IfFileName glob="*.zip"/>
<IfLastModified age="${log4j2.max.file.time}"/>
</Delete>
</DefaultRolloverStrategy>
</RollingRandomAccessFile>
</Appenders>
<Loggers>
<Root level="${log4j2.level}" includeLocation="false">
<AppenderRef ref="Console"/>
<AppenderRef ref="FILE_INFO"/>
<AppenderRef ref="FILE_ERROR"/>
</Root>
</Loggers>
</Configuration> |
Looking more carefully at your memory graphs, I think that the expansion of
Given the huge |
Hi @ppkarwasz
Does this refer to the fact that I only configured
However, with the same configuration and code in version 2.23.1, why didn’t I observe this memory increase? What specific code changes from 2.23.1 to 2.24.3 might have caused this memory behavior change? Thank you for your patient explanations. |
Apparently the behavior in To draw some conclusions from this thread, what are your expectations regarding GC-free mode?
|
@ppkarwasz, keeping the TCM size under a certain threshold sounds like a good thing, similar to what we do with |
Hi @ppkarwasz I think I’ve got it. Let me describe what happened again to ensure I fully understand. My program doesn’t specify a In version 2.23.1, logging-log4j2/log4j-core/src/main/java/org/apache/logging/log4j/core/LoggerContext.java Lines 493 to 495 in fea2a71
logging-log4j2/log4j-api/src/main/java/org/apache/logging/log4j/spi/AbstractLogger.java Lines 134 to 137 in fea2a71
In version 2.24.3, logging-log4j2/log4j-core/src/main/java/org/apache/logging/log4j/core/LoggerContext.java Lines 93 to 97 in 4aa5efe
logging-log4j2/log4j-core/src/main/java/org/apache/logging/log4j/core/LoggerContext.java Lines 538 to 540 in 4aa5efe
These two different MessageFactory implementations are the main reason I’ve observed a gradual increase in memory usage: to support GC-free,
Pre-allocating a suitably sized |
Description
I am using Log4j2 in my project and recently upgraded from version 2.23.1 to 2.24.3. The application employs an asynchronous logging strategy with Disruptor version 3.4.3. Below is the relevant configuration:
log4j2.contextSelector=org.apache.logging.log4j.core.async.AsyncLoggerContextSelector
log4j2.asyncQueueFullPolicy=org.apache.logging.log4j.core.async.DiscardingAsyncQueueFullPolicy
log4j2.discardThreshold=ERROR
log4j2.asyncLoggerWaitStrategy=Yield
log4j2.asyncLoggerRingBufferSize=1048576
The logging frequency in my application is relatively low, averaging about 10 log entries per second. When using version 2.23.1, memory usage remained stable based on monitoring data . However, after upgrading to 2.24.3, I observed a gradual increase in memory usage over time, which eventually stabilizes at a fixed threshold.

This behavior differs significantly from what I experienced with 2.23.1. Through heap dump analysis at different time intervals, I confirmed that the memory growth is primarily associated with
RingBufferLogEvent
objects within Log4j2’s RingBuffer implementation, with the RingBuffer’s memory footprint consistently increasing over time.I noticed that the 2.24.3 release includes several changes related to memory management. Could any of these changes, particularly those affecting the RingBuffer or
RingBufferLogEvent
, explain the behavior I am observing?If additional details are needed, please let me know, and I’d be happy to provide them.
Environment
The text was updated successfully, but these errors were encountered: