JAR फ़ाइल के बाहर गुण फ़ाइल पढ़ें


131

मेरे पास एक JAR फ़ाइल है जहाँ मेरे सभी कोड को चलाने के लिए संग्रहीत किया गया है। मुझे एक प्रॉपर्टी फाइल को एक्सेस करना होगा जिसे प्रत्येक रन से पहले बदलना / संपादित करना होगा। मैं गुण फ़ाइल को उसी निर्देशिका में रखना चाहता हूं जहाँ JAR फ़ाइल है। वैसे भी जावा को उस निर्देशिका से गुण फ़ाइल लेने के लिए कहना है?

नोट: मैं संपत्तियों की फ़ाइल को होम डायरेक्टरी में नहीं रखना चाहता या कमांड लाइन के तर्क में प्रॉपर्टीज़ फ़ाइल का मार्ग पास नहीं करना चाहता।


2
इस उत्तर को देखें - "इसके बजाय जार के अंदर 'डिफ़ॉल्ट' फ़ाइल संग्रहीत करें। यदि इसे बदल दिया जाता है, तो परिवर्तित फ़ाइल को किसी अन्य स्थान पर संग्रहीत करें। एक सामान्य स्थान एक उप-निर्देशिका है user.home। फ़ाइल की जाँच करते समय, पहले अस्तित्व की जाँच करें। फ़ाइल सिस्टम पर एक परिवर्तित फ़ाइल, और यदि यह मौजूद नहीं है, तो डिफ़ॉल्ट फ़ाइल लोड करें। " BTW "मुझे नहीं चाहिए .." जो आप चाहते हैं वह कम महत्वपूर्ण है जो काम करता है और व्यावहारिक है। स्टोरिंग ऐप। आवेदन निर्देशिका में सेटिंग्स ओरेकल और एमएस (और शायद अन्य) दोनों द्वारा दृढ़ता से हतोत्साहित की जाती हैं।
एंड्रयू थॉम्पसन

3
गुणों को जार निर्देशिका में रखने की आवश्यकता के कारण यह है कि जब पूरी निर्देशिका (जार और संपत्ति सहित) को किसी अन्य मशीन में कॉपी किया जाता है तो उन्हें एक साथ रखना बेहतर होता है।
नील

और अगर मैं उपयोगकर्ता को संपत्ति फ़ाइल पथ को पारित करने के लिए मजबूर करता हूं तो उसे हर बार उसे बदलने की ज़रूरत होती है जब वह एक अलग मशीन से बैच फ़ाइल चलाता है।
नील

जवाबों:


144

इसलिए, आप अपनी .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();
}

7
यह समाधान काम करता है। सटीक आवश्यकता और विस्तृत कोड को समझने के लिए धन्यवाद। मैंने सत्यापित किया कि गुण फ़ाइल जार फ़ाइल के अंदर नहीं है, लेकिन फिर भी यह उसी निर्देशिका से फ़ाइल तक पहुंच सकता है जहां जार फ़ाइल है। इस तरह से पूर्ण पथ की आवश्यकता नहीं है कठिन कोड की आवश्यकता है। जार और प्रॉपर्टी फ़ाइल दोनों को अब किसी भी निर्देशिका में कॉपी किया जा सकता है और स्वतंत्र रूप से चलाया जा सकता है।
नील

3
यदि आप पूर्व से आदेश को निष्पादित करते हैं तो फ़ाइल नहीं मिलेगी: {{java -jar build / main.jar}}। क्या आपके पास उसके लिए कोई फिक्स है, @eee?
डारियन

@ डारियन यहाँ ठीक करने के लिए कुछ भी नहीं है; यह केवल उसी रूप में काम करता है, जहां जार और गुण फ़ाइल उसी ./रूट फ़ोल्डर (समान निर्देशिका स्तर) पर होनी चाहिए, जैसा कि मैंने फ़ाइल संगठन आर्किटेक्चर में वर्णित किया है। (मूल पोस्टर द्वारा निर्धारित आवश्यकता के अनुसार)
चाचा

@Darian, इसलिए यदि आप निष्पादित करना चाहते हैं, तो आपको java -jar build/main.jarगुण फ़ाइल को buildफ़ोल्डर में रखने की आवश्यकता है ताकि यह उसी निर्देशिका स्तर पर जार के रूप में हो।
चाचा

7
आपकी प्रतिक्रिया @eee के लिए धन्यवाद, समस्या यह है कि मुझे नहीं पता कि उपयोगकर्ता कहां निष्पादित करेगा java -jar path/to/jar/file। लेकिन मुझे एक और प्रश्न में समाधान मिला:String path = ClassLoader.getSystemClassLoader().getResource(".").getPath() + "/main.properties";
डारियन

42

मैंने इसे दूसरे तरीके से किया।

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. जार फ़ाइल पथ प्राप्त करें।
  2. उस फ़ाइल का मूल फ़ोल्डर प्राप्त करें।
  3. अपने पथ फ़ाइल नाम के साथ InputStreamPath में उस पथ का उपयोग करें।

मुझे getParentFile () भाग को छोड़ना पड़ा, इसलिए मैंने इसका उपयोग किया: स्ट्रिंग गुणपद = jarPath.getAbsolutePath (); लेकिन यह सब इस बात पर निर्भर करता है कि फ़ाइल कहाँ स्थित है
MobileMon

4
बस बदलें "jarPath.getParentFile ()। GetAbsolutePath ();" "jarPath.getParent ()" के लिए। तब एक आकर्षण की तरह काम करता है।
स्टैकएडक्ट

1
मेरे मामले में भी यही समस्या है लेकिन मेरे पास स्प्रिंग बेस प्रोजेक्ट है। वसंत को कैसे बताएं कि फ़ाइल जार फ़ाइल के पास है? किसी भी विचार
Mubasher

3

जार फ़ाइल से आपकी फ़ाइल निर्देशिका पर फ़ाइलों तक पहुँचने में हमेशा कोई समस्या होती है। एक जार फ़ाइल में classpath प्रदान करना बहुत सीमित है। इसके बजाय अपने प्रोग्राम को शुरू करने के लिए एक बैट फाइल या एक श फाइल का उपयोग करें। इस तरह आप सिस्टम पर कहीं भी किसी भी फ़ोल्डर को संदर्भित करते हुए, अपनी पसंद के अनुसार अपने वर्गपथ को निर्दिष्ट कर सकते हैं।

इस प्रश्न पर मेरे उत्तर की भी जाँच करें:

जावा परियोजना के लिए। exe फ़ाइल जिसमें साइक्लाइट है


1

मेरे पास एक समान मामला है: मेरी *.jarफ़ाइल को उक्त *.jarफ़ाइल के बगल में एक निर्देशिका में फ़ाइल तक पहुंचने की इच्छा है इस संदर्भ में भी देखें ।

मेरी फ़ाइल संरचना है:

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

मैं निम्नलिखित के साथ फ़ाइल के some.txtअंदर एक इनपुटस्ट्रीम ( File) लोड करने में सक्षम हूं *.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

मेरे पास 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

0

यह मेरे लिए काम करता है। से अपने गुण फ़ाइल लोड करें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

यहां यदि आप उल्लेख करते हैं .getPath()तो वह जार का रास्ता लौटाएगा और मुझे लगता है कि आपको जार के साथ रखी गई अन्य सभी कॉन्फिग फाइलों को संदर्भित करने के लिए उसके अभिभावक की आवश्यकता होगी। यह कोड विंडोज पर काम करता है। मुख्य वर्ग के भीतर कोड जोड़ें।

File jarDir = new File(MyAppName.class.getProtectionDomain().getCodeSource().getLocation().getPath());
String jarDirpath = jarDir.getParent();

System.out.println(jarDirpath);
हमारी साइट का प्रयोग करके, आप स्वीकार करते हैं कि आपने हमारी Cookie Policy और निजता नीति को पढ़ और समझा लिया है।
Licensed under cc by-sa 3.0 with attribution required.