读取 JAR 文件外的属性文件
Posted
技术标签:
【中文标题】读取 JAR 文件外的属性文件【英文标题】:Read properties file outside JAR file 【发布时间】:2012-02-05 05:07:33 【问题描述】:我有一个 JAR 文件,我的所有代码都在其中存档以供运行。我必须访问一个需要在每次运行之前更改/编辑的属性文件。我想将属性文件保存在 JAR 文件所在的同一目录中。有没有办法告诉 Java 从该目录中获取属性文件?
注意:我不想将属性文件保留在主目录中,也不想在命令行参数中传递属性文件的路径。
【问题讨论】:
参见this answer - “而是将'default'文件存储在Jar中。如果更改了,请将更改后的文件存储在另一个位置。一个常见的位置是user.home
。检查文件时,首先检查文件系统上是否存在修改过的文件,如果不存在则加载默认文件。" BTW “我不想要..” 你想要的不如有效的和实用的重要。存储应用程序。 Oracle 和 MS(可能还有其他人)都强烈反对应用程序目录中的设置。
我之所以需要将properties文件保留在jar目录中,是因为在将整个目录(包括jar和property)复制到另一台机器运行时,最好将它们保留在一起。
如果我强制用户传递属性文件路径,那么他每次从不同的机器运行批处理文件时都需要更改它。
【参考方案1】:
因此,您希望将与 main/runnable jar 位于同一文件夹中的 .properties
文件视为文件而不是 main/runnable jar 的资源。那样的话,我自己的解决方法如下:
第一件事:你的程序文件架构应该是这样的(假设你的主程序是main.jar,它的主要属性文件是main.properties):
./ - the root of your program
|__ main.jar
|__ main.properties
使用此架构,您可以在 main.jar 运行之前或期间使用任何文本编辑器修改 main.properties 文件中的任何属性(取决于程序的当前状态),因为它只是一个基于文本的文件.例如,您的 main.properties 文件可能包含:
app.version=1.0.0.0
app.name=Hello
因此,当您从根/基本文件夹运行主程序时,通常您会像这样运行它:
java -jar ./main.jar
或者,直接:
java -jar main.jar
在您的 main.jar 中,您需要为 main.properties 文件中的每个属性创建一些实用方法;假设app.version
属性将具有getAppVersion()
方法,如下所示:
/**
* Gets the app.version property value from
* the ./main.properties file of the base folder
*
* @return app.version string
* @throws IOException
*/
import java.util.Properties;
public static String getAppVersion() throws IOException
String versionString = null;
//to load application's properties, we use this class
Properties mainProperties = new Properties();
FileInputStream file;
//the base folder is ./, the root of the main.properties file
String path = "./main.properties";
//load the file handle for main.properties
file = new FileInputStream(path);
//load all the properties from this file
mainProperties.load(file);
//we have loaded the properties, so close the file handle
file.close();
//retrieve the property we are intrested, the app.version
versionString = mainProperties.getProperty("app.version");
return versionString;
在主程序的任何需要app.version
值的部分,我们调用它的方法如下:
String version = null;
try
version = getAppVersion();
catch (IOException ioe)
ioe.printStackTrace();
【讨论】:
此解决方案有效。感谢您了解确切的要求和详细的代码。我验证了属性文件不在 jar 文件中,但它仍然可以从 jar 文件所在的同一目录访问该文件。这样就不需要绝对路径硬代码。 jar 和属性文件现在都可以复制到任何目录并独立运行。 如果从外部执行命令将找不到该文件,例如:java -jar build/main.jar。你有什么解决办法吗,@eee? @Darian 这里没有什么可修复的;它仅在 jar 和属性文件必须与我在文件组织架构中描述的相同的./
根文件夹(相同的目录级别)上的情况下按设计工作。 (按原发帖人要求)
@Darian,所以如果你想执行java -jar build/main.jar
,你需要把属性文件放在build
文件夹中,这样它就和jar在同一目录级别。跨度>
感谢您的回复@eee,问题是我不知道用户将在哪里执行java -jar path/to/jar/file
。但我在另一个问题中找到了解决方案:String path = ClassLoader.getSystemClassLoader().getResource(".").getPath() + "/main.properties";
【参考方案2】:
我是通过其他方式做到的。
Properties prop = new Properties();
try
File jarPath=new File(MyClass.class.getProtectionDomain().getCodeSource().getLocation().getPath());
String propertiesPath=jarPath.getParentFile().getAbsolutePath();
System.out.println(" propertiesPath-"+propertiesPath);
prop.load(new FileInputStream(propertiesPath+"/importer.properties"));
catch (IOException e1)
e1.printStackTrace();
-
获取 Jar 文件路径。
获取该文件的父文件夹。
在 InputStreamPath 中将该路径与您的属性文件名一起使用。
【讨论】:
我不得不删除 getParentFile() 部分,所以我改用: String propertiesPath=jarPath.getAbsolutePath();但这一切都取决于文件所在的位置 只需替换“jarPath.getParentFile().getAbsolutePath();”到“jarPath.getParent()”。然后像魅力一样工作。 同样是我的问题,但我有弹簧基础项目。如何告诉spring该文件在jar文件旁边?任何想法【参考方案3】:从 jar 文件访问文件目录中的文件总是有问题。在 jar 文件中提供类路径非常有限。而是尝试使用 bat 文件或 sh 文件来启动您的程序。通过这种方式,您可以随意指定类路径,引用系统上任何位置的任何文件夹。
同时查看我对这个问题的回答:
making .exe file for java project containing sqlite
【讨论】:
【参考方案4】:我有一个类似的情况:希望我的*.jar
文件访问所述*.jar
文件旁边的目录中的文件。也请参考THIS ANSWER。
我的文件结构是:
./ - the root of your program
|__ *.jar
|__ dir-next-to-jar/some.txt
我可以使用以下内容将文件(例如,some.txt
)加载到 *.jar
文件中的 InputStream:
InputStream stream = null;
try
stream = ThisClassName.class.getClass().getResourceAsStream("/dir-next-to-jar/some.txt");
catch(Exception e)
System.out.print("error file to stream: ");
System.out.println(e.getMessage());
然后用stream
做任何你想做的事情
【讨论】:
【参考方案5】:我有一个通过类路径或使用 log4j2.properties 从外部配置执行的示例
package org.mmartin.app1;
import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.util.Properties;
import org.apache.logging.log4j.Logger;
import org.apache.logging.log4j.core.LoggerContext;
import org.apache.logging.log4j.LogManager;
public class App1
private static Logger logger=null;
private static final String LOG_PROPERTIES_FILE = "config/log4j2.properties";
private static final String CONFIG_PROPERTIES_FILE = "config/config.properties";
private Properties properties= new Properties();
public App1()
System.out.println("--Logger intialized with classpath properties file--");
intializeLogger1();
testLogging();
System.out.println("--Logger intialized with external file--");
intializeLogger2();
testLogging();
public void readProperties()
InputStream input = null;
try
input = new FileInputStream(CONFIG_PROPERTIES_FILE);
this.properties.load(input);
catch (IOException e)
logger.error("Unable to read the config.properties file.",e);
System.exit(1);
public void printProperties()
this.properties.list(System.out);
public void testLogging()
logger.debug("This is a debug message");
logger.info("This is an info message");
logger.warn("This is a warn message");
logger.error("This is an error message");
logger.fatal("This is a fatal message");
logger.info("Logger's name: "+logger.getName());
private void intializeLogger1()
logger = LogManager.getLogger(App1.class);
private void intializeLogger2()
LoggerContext context = (org.apache.logging.log4j.core.LoggerContext) LogManager.getContext(false);
File file = new File(LOG_PROPERTIES_FILE);
// this will force a reconfiguration
context.setConfigLocation(file.toURI());
logger = context.getLogger(App1.class.getName());
public static void main(String[] args)
App1 app1 = new App1();
app1.readProperties();
app1.printProperties();
--Logger intialized with classpath properties file--
[DEBUG] 2018-08-27 10:35:14.510 [main] App1 - This is a debug message
[INFO ] 2018-08-27 10:35:14.513 [main] App1 - This is an info message
[WARN ] 2018-08-27 10:35:14.513 [main] App1 - This is a warn message
[ERROR] 2018-08-27 10:35:14.513 [main] App1 - This is an error message
[FATAL] 2018-08-27 10:35:14.513 [main] App1 - This is a fatal message
[INFO ] 2018-08-27 10:35:14.514 [main] App1 - Logger's name: org.mmartin.app1.App1
--Logger intialized with external file--
[DEBUG] 2018-08-27 10:35:14.524 [main] App1 - This is a debug message
[INFO ] 2018-08-27 10:35:14.525 [main] App1 - This is an info message
[WARN ] 2018-08-27 10:35:14.525 [main] App1 - This is a warn message
[ERROR] 2018-08-27 10:35:14.525 [main] App1 - This is an error message
[FATAL] 2018-08-27 10:35:14.525 [main] App1 - This is a fatal message
[INFO ] 2018-08-27 10:35:14.525 [main] App1 - Logger's name: org.mmartin.app1.App1
-- listing properties --
dbpassword=password
database=localhost
dbuser=user
【讨论】:
【参考方案6】:如果你提到.getPath()
,那么这将返回 Jar 的路径,我猜
您将需要其父级来引用与 jar 一起放置的所有其他配置文件。
此代码适用于 Windows。在主类中添加代码。
File jarDir = new File(MyAppName.class.getProtectionDomain().getCodeSource().getLocation().getPath());
String jarDirpath = jarDir.getParent();
System.out.println(jarDirpath);
【讨论】:
【参考方案7】:这对我有用。从current directory
加载您的属性文件。
注意:方法Properties#load
uses ISO-8859-1 encoding。
Properties properties = new Properties();
properties.load(new FileReader(new File(".").getCanonicalPath() + File.separator + "java.properties"));
properties.forEach((k, v) ->
System.out.println(k + " : " + v);
);
确保 java.properties
位于 current directory
。你可以写一个小启动脚本,切换到之前的正确目录,比如
#! /bin/bash
scriptdir="$( cd "$( dirname "$BASH_SOURCE[0]" )" && pwd )"
cd $scriptdir
java -jar MyExecutable.jar
cd -
在您的项目中,只需将java.properties
文件放在您的项目根目录中,以便让此代码也可以在您的 IDE 中运行。
【讨论】:
【参考方案8】:File parentFile = new File(".");
String parentPath = file.getCanonicalPath();
File resourceFile = new File(parentPath+File.seperator+"<your config file>");
【讨论】:
请在您的回答中添加一些解释。以上是关于读取 JAR 文件外的属性文件的主要内容,如果未能解决你的问题,请参考以下文章
java 如何读取jar包外的properties文件(转)
如何读取jar包外的properties文件和log4j.properties