
本文旨在解决使用 jpackage 打包 java 应用程序为 windows exe 后,log4j2 日志系统无法正常工作的问题,尤其是在日志路径依赖于应用程序根目录的自定义系统属性时。核心问题在于 log4j2 记录器(logger)的初始化时机与日志配置重载操作之间的冲突。通过延迟记录器的实例化,确保在配置完全加载并重载之后才获取记录器实例,可以有效解决此问题,从而使日志文件按预期写入应用程序的指定目录。
在开发 Java 应用程序时,日志功能是不可或缺的一部分。Log4j2 作为一个功能强大的日志框架,常被用于管理应用程序的日志输出。然而,当使用 jpackage 工具将 Java 应用程序打包成 Windows 可执行文件(EXE)后,可能会遇到一个棘手的问题:应用程序在直接运行 JAR 包时日志功能一切正常,但运行生成的 EXE 文件时却无法创建日志文件或日志文件创建位置不正确。
具体表现为,如果日志配置中依赖于一个自定义的系统属性(例如 app.root)来指定日志文件的存放路径,当该属性在 main 方法中设置并尝试重载 Log4j2 配置后,EXE 版本依然无法正确解析路径,导致日志文件被创建在应用程序的当前工作目录,而非预期的应用程序根目录。
本教程将以 Java 17、Log4j2 2.19.0、SLF4J 2.0.4 和 Maven Shade Plugin 3.2.4 为例,深入探讨此问题并提供一个可靠的解决方案。
为了实现日志文件存放在应用程序根目录,通常会在 log4j.properties 文件中定义一个使用系统属性的路径,例如:
# Root logger option
log4j.rootLogger=INFO, DEBUG, file, stdout
# Direct log messages to a log file
log4j.appender.file=org.apache.log4j.RollingFileAppender
log4j.appender.file.File=${app.root}/application.log
log4j.appender.file.MaxFileSize=200KB
log4j.appender.file.MaxBackupIndex=4
log4j.appender.file.layout.ConversionPattern=%i %-6p [%-30c{1} :%4L] %m%n
log4j.appender.file.layout=MyPatternLayoutWithQualifiedPath
# Direct log messages to stdout
log4j.appender.stdout=org.apache.log4j.ConsoleAppender
log4j.appender.stdout.Target=System.out
log4j.appender.stdout.layout.ConversionPattern=%i %-6p [%-30c{1} :%4L] %m%n
log4j.appender.stdout.layout=MyPatternLayoutWithQualifiedPath在应用程序的 main 方法中,我们会获取应用程序的根目录,并将其设置为 app.root 系统属性,然后重载 Log4j2 配置,以确保新的日志路径生效:
import org.apache.logging.log4j.core.LoggerContext;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.io.File;
import java.net.URL;
public class Main {
// 问题所在:静态Logger在main方法执行前初始化
private static final Logger log = LoggerFactory.getLogger(Main.class);
public static void main(String[] args) {
// 启用 Log4j1 兼容模式
System.setProperty("log4j1.compatibility", "true");
// 获取应用程序根目录并设置 app.root 系统属性
URL mySource = Main.class.getProtectionDomain().getCodeSource().getLocation();
File applicationRootDirectory = new File(new File(mySource.getPath()).getParent());
System.setProperty("app.root", applicationRootDirectory.getAbsolutePath().replaceAll("%20", " "));
// 重载 Log4j2 配置
LoggerContext.getContext(false).reconfigure();
// 此处开始应用程序的其余逻辑,log 此时可能已经初始化完毕
log.info("应用程序启动...");
// ... rest of Main class
}
}使用 jpackage 命令打包:
jpackage.exe ^ --name "the name of the app" ^ --app-version %version% ^ --vendor "vendor name" ^ --icon "icon.ico" ^ --license-file license.txt ^ --file-associations file-association.properties ^ --input input_directory ^ --main-jar application.jar ^ --main-class path.to.MainClass ^ --type exe ^ --win-per-user-install ^ --win-dir-chooser ^ --win-menu ^ --win-menu-group menuGroupName ^ --win-shortcut
在上述配置和代码下,直接运行 application.jar 时,日志文件会正确地创建在应用程序根目录。然而,运行 jpackage 生成的 application.exe 时,日志文件却无法在预期位置生成,反而可能出现在当前工作目录。
问题的根本原因在于 Log4j2 记录器(Logger)的初始化时机。当我们将 Logger 声明为类的静态字段时:
private static final Logger log = LoggerFactory.getLogger(Main.class);
这个静态字段会在 Main 类加载时被初始化。类加载通常发生在 main 方法执行之前。这意味着,在 main 方法中设置 app.root 系统属性并调用 LoggerContext.getContext(false).reconfigure() 来重载配置之前,log 实例就已经被创建了。
当 log 实例首次创建时,Log4j2 会尝试加载其配置。如果此时 app.root 系统属性尚未设置,或者其值不正确,Log4j2 会使用默认值或解析失败,从而导致日志路径不正确。即使随后在 main 方法中设置了 app.root 并重载了配置,对于已经初始化过的 log 实例,可能无法完全生效,或者在某些环境下(如 jpackage 生成的 EXE)行为表现不一致。
解决此问题的关键是确保在所有必要的系统属性设置完毕并 Log4j2 配置重载完成后,再获取并初始化 Logger 实例。这意味着应该将 Logger 的初始化从静态字段移到 main 方法内部,在 reconfigure() 调用之后。
修改后的 main 方法如下:
import org.apache.logging.log4j.core.LoggerContext;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.io.File;
import java.net.URL;
public class Main {
// 不再在类级别声明静态Logger
// private static final Logger log = LoggerFactory.getLogger(Main.class);
public static void main(String[] args) {
// 启用 Log4j1 兼容模式
System.setProperty("log4j1.compatibility", "true");
// 获取应用程序根目录并设置 app.root 系统属性
URL mySource = Main.class.getProtectionDomain().getCodeSource().getLocation();
File applicationRootDirectory = new File(new File(mySource.getPath()).getParent());
System.setProperty("app.root", applicationRootDirectory.getAbsolutePath().replaceAll("%20", " "));
// 重载 Log4j2 配置
// 确保在获取任何Logger实例之前完成配置
LoggerContext.getContext(false).reconfigure();
// 延迟Logger的初始化,确保它在配置完全生效后才被创建
Logger log = LoggerFactory.getLogger(Main.class);
// 此处开始应用程序的其余逻辑
log.info("应用程序启动成功,日志已配置。");
// ... rest of Main class
}
}通过将 Logger log = LoggerFactory.getLogger(Main.class); 这一行移到 LoggerContext.getContext(false).reconfigure(); 之后,我们确保了:
当使用 jpackage 将 Java 应用程序打包为 EXE 后,如果遇到 Log4j2 日志功能异常,特别是日志路径依赖于动态设置的系统属性时,很可能是因为记录器(Logger)的初始化时机早于日志配置的重载。通过将 Logger 实例的获取操作延迟到 main 方法中,在所有必要的系统属性设置和 LoggerContext.getContext(false).reconfigure() 调用之后,可以确保 Logger 实例在正确的配置环境下被初始化。这一简单的调整能够有效解决 EXE 版本中日志无法按预期工作的难题,保证应用程序日志输出的稳定性和正确性。
以上就是使用 jpackage 打包后 Log4j2 日志失效的解决方案的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号