Чтение файла свойств вне файла JAR - PullRequest
118 голосов
/ 08 января 2012

У меня есть JAR-файл, в котором весь мой код заархивирован для работы.Мне нужно получить доступ к файлу свойств, который нужно менять / редактировать перед каждым запуском.Я хочу сохранить файл свойств в том же каталоге, где находится файл JAR.Можно ли в любом случае сказать Java забрать файл свойств из этого каталога?

Примечание. Я не хочу сохранять файл свойств в домашнем каталоге или передавать путь файла свойств в аргументе командной строки.

Ответы [ 6 ]

135 голосов
/ 08 января 2012

Итак, вы хотите рассматривать ваш файл .properties в той же папке, что и основной / исполняемый файл, как файл, а не как ресурс основного / исполняемого файла. В этом случае мое собственное решение выглядит следующим образом:

Перво-наперво: архитектура вашего программного файла должна быть такой (при условии, что ваша основная программа - main.jar, а ее файл основных свойств - main.properties):

./ - the root of your program
 |__ main.jar
 |__ main.properties

С помощью этой архитектуры вы можете изменять любое свойство в файле main.properties с помощью любого текстового редактора до или во время работы вашего main.jar (в зависимости от текущего состояния программы), так как это всего лишь текстовый файл. , Например, ваш файл 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();
}
35 голосов
/ 15 января 2014

Я сделал это другим способом.

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();
    }
  1. Получить путь к файлу Jar.
  2. Получить родительскую папку этого файла.
  3. Используйте этот путь в InputStreamPath вместе с именем файла свойств.
2 голосов
/ 08 января 2012

Всегда существует проблема доступа к файлам в вашем каталоге файлов из файла JAR.Предоставление пути к классам в jar-файле очень ограничено.Вместо этого попробуйте использовать bat-файл или sh-файл для запуска вашей программы.Таким образом, вы можете указать свой путь к классу в любом случае, ссылаясь на любую папку в любом месте системы.

Также проверьте мой ответ на этот вопрос:

создание файла .exe для проекта Javaсодержащие sqlite

1 голос
/ 22 января 2017

У меня похожий случай: я хочу, чтобы мой *.jar файл имел доступ к файлу в каталоге рядом с указанным *.jar файлом.См. Также ЭТОТ ОТВЕТ .

Моя файловая структура:

./ - the root of your program
|__ *.jar
|__ dir-next-to-jar/some.txt

Я могу загрузить файл (скажем, some.txt) вInputStream внутри файла *.jar со следующим:

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

0 голосов
/ 19 февраля 2019

Это работает для меня.Загрузите файл свойств из current directory

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 также.

0 голосов
/ 27 августа 2018

У меня есть пример работы как по classpath, так и из внешнего конфига с 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
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...