轉載:https://blog.csdn.net/ThinkWon/article/details/95043111
Apache Log4j2詳解
文章目錄
簡介
Apache Log4j 2是對Log4j的升級,它比其前身Log4j 1.x提供了重大改進,並提供了Logback中可用的許多改進,同時修復了Logback架構中的一些問題。是目前最優秀的Java日誌框架,沒有之一。
特徵
API分離
Log4j的API與實現分開,使應用程序開發人員可以清楚地瞭解可以使用哪些類和方法,同時確保向前兼容性。這使Log4j團隊能夠以安全且兼容的方式進行改進。
性能提升
Log4j 2包含基於LMAX Disruptor庫的下一代異步記錄器。在多線程場景中,異步記錄器的吞吐量比Log4j 1.x和Logback高18倍,延遲低。
自動重新加載配置
與Logback一樣,Log4j 2可以在修改時自動重新加載其配置。與Logback不同,它會在重新配置發生時不會丟失日誌事件。
高級過濾
與Logback一樣,Log4j 2支持基於Log事件中的上下文數據,標記,正則表達式和其他組件進行過濾。此外,過濾器還可以與記錄器關聯。與Logback不同,您可以在任何這些情況下使用通用的Filter類。
插件架構
Log4j使用插件模式配置組件。因此,您無需編寫代碼來創建和配置Appender,Layout,Pattern Converter等。在配置了的情況下,Log4j自動識別插件並使用它們。
無垃圾機制
在穩態日誌記錄期間,Log4j 2 在獨立應用程序中是無垃圾的,在Web應用程序中是低垃圾。這減少了垃圾收集器的壓力,並且可以提供更好的響應性能。
使用Log4j2
引用依賴
在一般項目中使用Log4j2至少需要引用log4j-api-2.x和log4j-core-2.x這兩個jar包。
<dependency>
<groupId>org.apache.logging.log4j</groupId>
<artifactId>log4j-core</artifactId>
<version>2.12.0</version>
</dependency>
<dependency>
<groupId>org.apache.logging.log4j</groupId>
<artifactId>log4j-api</artifactId>
<version>2.12.0</version>
</dependency>
在spring boot項目中使用Log4j2
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
<!-- 排除spring boot默認日誌logback -->
<exclusions>
<exclusion>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-logging</artifactId>
</exclusion>
</exclusions>
</dependency>
<!-- 引入log4j2依賴 -->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-log4j2</artifactId>
</dependency>
小編在使用中遇到一些坑,同時引入spring-boot-starter和spring-boot-starter-web依賴時,需要在spring-boot-starter中排除spring-boot-starter-logging依賴,因爲spring-boot-starter-web依賴於spring-boot-starter,根據Maven依賴關係,在spring-boot-starter中排除spring-boot-starter-logging依賴才能排除成功。
添加配置文件
默認情況下,Log4j2在classpath下查找名爲log4j2.xml的配置文件。你也可以使用Java啓動命令指定配置文件的全路徑。-Dlog4j.configurationFile=opt/demo/log4j2.xml,你還可以使用Java代碼指定配置文件路徑
import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.core.LoggerContext;
import java.io.File;
public static void main(String[] args) {
LoggerContext loggerContext = (LoggerContext) LogManager.getContext(false);
File file = new File("opt/demo/log4j2.xml");
loggerContext.setConfigLocation(file.toURI());
}
同步日誌,混合日誌和異步日誌配置詳解
Log4j2同步日誌,混合日誌和異步日誌配置詳解
配置文件詳解
Log4j2配置文件詳解
日誌重複打印問題
如果Root中的日誌包含了Logger中的日誌信息,並且AppenderRef是一樣的配置,則日誌會打印兩次。
這是log4j2繼承機制問題,在Log4j2中,logger是有繼承關係的,root是根節點,在log4j2中,有個additivity的屬性,它是子Logger 是否繼承 父Logger 的 輸出源(appender) 的屬性。具體說,默認情況下子Logger會繼承父Logger的appender,也就是說子Logger會在父Logger的appender裏輸出。若是additivity設爲false,則子Logger只會在自己的appender裏輸出,而不會在父Logger的appender裏輸出。
要打破這種傳遞性,也非常簡單,在logger中添加 additivity = “false”,如下所示:
<?xml version="1.0" encoding="UTF-8"?>
<Configuration>
<Appenders>
<Console name="Console">
<PatternLayout>
<Pattern>%d{yyyy-MM-dd HH:mm:ss,SSS} [%t] %-5level %logger{0} - %msg%n</Pattern>
</PatternLayout>
</Console>
</Appenders>
<Loggers>
<!-- name屬性爲項目包名或者類名 -->
<Logger name="com.jourwon" level="debug" additivity="false">
<AppenderRef ref="Console"/>
</Logger>
<Root level="error">
<AppenderRef ref="Console"/>
</Root>
</Loggers>
</Configuration>
使用Lombok工具簡化創建Logger類
lombok就是一個註解工具jar包,能幫助我們省略一繁雜的代碼。具體介紹可以看我的博客Lombok簡介、使用、工作原理、優缺點
使用Lombok後,@Slf4j註解生成了log日誌常量,無需去聲明一個log就可以在類中使用log記錄日誌。
@Slf4j
public class Log4jTest {
public static void main(String[] args) {
log.error("Something else is wrong here");
}
}