जार के रूप में चलने पर क्लासपाथ संसाधन नहीं मिला


128

स्प्रिंग बूट १.१.५ और १.१.६ दोनों में इस समस्या के होने पर - मैं @Value एनोटेशन का उपयोग करते हुए एक क्लासपैथ संसाधन लोड कर रहा हूं, जो एसटीएस (३.६.०, विंडोज) के भीतर से एप्लिकेशन चलाने पर ठीक काम करता है। हालाँकि, जब मैं एक mvan पैकेज चलाता हूं और तब जार को चलाने का प्रयास करता हूं, तो मुझे FileNotFound अपवाद मिलता है।

संसाधन, message.txt, src / main / resource में है। मैंने जार का निरीक्षण किया है और सत्यापित किया है कि इसमें शीर्ष स्तर पर फ़ाइल "message.txt" शामिल है (Application.properties के समान स्तर)।

ये रहा आवेदन:

@Configuration
@ComponentScan
@EnableAutoConfiguration
public class Application implements CommandLineRunner {

    private static final Logger logger = Logger.getLogger(Application.class);

    @Value("${message.file}")
    private Resource messageResource;

    public static void main(String[] args) {
        SpringApplication.run(Application.class, args);
    }

    @Override
    public void run(String... arg0) throws Exception {
        // both of these work when running as Spring boot app from STS, but
        // fail after mvn package, and then running as java -jar
        testResource(new ClassPathResource("message.txt"));
        testResource(this.messageResource);
    }

    private void testResource(Resource resource) {
        try {
            resource.getFile();
            logger.debug("Found the resource " + resource.getFilename());
        } catch (IOException ex) {
            logger.error(ex.toString());
        }
    }
}

अपवाद:

c:\Users\glyoder\Documents\workspace-sts-3.5.1.RELEASE\classpath-resource-proble
m\target>java -jar demo-0.0.1-SNAPSHOT.jar

  .   ____          _            __ _ _
 /\\ / ___'_ __ _ _(_)_ __  __ _ \ \ \ \
( ( )\___ | '_ | '_| | '_ \/ _` | \ \ \ \
 \\/  ___)| |_)| | | | | || (_| |  ) ) ) )
  '  |____| .__|_| |_|_| |_\__, | / / / /
 =========|_|==============|___/=/_/_/_/
 :: Spring Boot ::        (v1.1.5.RELEASE)

2014-09-16 08:46:34.635  INFO 5976 --- [           main] demo.Application
                  : Starting Application on 8W59XV1 with PID 5976 (C:\Users\glyo
der\Documents\workspace-sts-3.5.1.RELEASE\classpath-resource-problem\target\demo
-0.0.1-SNAPSHOT.jar started by glyoder in c:\Users\glyoder\Documents\workspace-s
ts-3.5.1.RELEASE\classpath-resource-problem\target)
2014-09-16 08:46:34.640 DEBUG 5976 --- [           main] demo.Application
                  : Running with Spring Boot v1.1.5.RELEASE, Spring v4.0.6.RELEA
SE
2014-09-16 08:46:34.681  INFO 5976 --- [           main] s.c.a.AnnotationConfigA
pplicationContext : Refreshing org.springframework.context.annotation.Annotation
ConfigApplicationContext@1c77b086: startup date [Tue Sep 16 08:46:34 EDT 2014];
root of context hierarchy
2014-09-16 08:46:35.196  INFO 5976 --- [           main] o.s.j.e.a.AnnotationMBe
anExporter        : Registering beans for JMX exposure on startup
2014-09-16 08:46:35.210 ERROR 5976 --- [           main] demo.Application
                  : java.io.FileNotFoundException: class path resource [message.
txt] cannot be resolved to absolute file path because it does not reside in the
file system: jar:file:/C:/Users/glyoder/Documents/workspace-sts-3.5.1.RELEASE/cl
asspath-resource-problem/target/demo-0.0.1-SNAPSHOT.jar!/message.txt
2014-09-16 08:46:35.211 ERROR 5976 --- [           main] demo.Application
                  : java.io.FileNotFoundException: class path resource [message.
txt] cannot be resolved to absolute file path because it does not reside in the
file system: jar:file:/C:/Users/glyoder/Documents/workspace-sts-3.5.1.RELEASE/cl
asspath-resource-problem/target/demo-0.0.1-SNAPSHOT.jar!/message.txt
2014-09-16 08:46:35.215  INFO 5976 --- [           main] demo.Application
                  : Started Application in 0.965 seconds (JVM running for 1.435)

2014-09-16 08:46:35.217  INFO 5976 --- [       Thread-2] s.c.a.AnnotationConfigA
pplicationContext : Closing org.springframework.context.annotation.AnnotationCon
figApplicationContext@1c77b086: startup date [Tue Sep 16 08:46:34 EDT 2014]; roo
t of context hierarchy
2014-09-16 08:46:35.218  INFO 5976 --- [       Thread-2] o.s.j.e.a.AnnotationMBe
anExporter        : Unregistering JMX-exposed beans on shutdown

जवाबों:


201

resource.getFile()संसाधन को स्वयं फ़ाइल सिस्टम पर उपलब्ध होने की उम्मीद है, अर्थात इसे जार फ़ाइल के अंदर नेस्टेड नहीं किया जा सकता है। यही कारण है कि यह तब काम करता है जब आप एसटीएस में अपना आवेदन चलाते हैं, लेकिन आपके आवेदन के निर्माण और निष्पादन योग्य जार से चलाने के बाद यह काम नहीं करता है। getFile()संसाधन की सामग्री का उपयोग करने के बजाय , मैं उपयोग करने की सलाह दूंगाgetInputStream() बजाय। यह आपको संसाधन की सामग्री को पढ़ने की अनुमति देगा जहां यह स्थित है।


6
मेरे मामले में, मुझे कीस्टोर फ़ाइल (एक टॉमकैट https कनेक्टर के लिए) के लिए एक पथ प्रदान करने की आवश्यकता है। मैं अपने जार के अंदर jks फ़ाइल लपेटना चाहता हूं। उपरोक्त समाधान के अनुसार यह संभव नहीं है। क्या आप इसे करने के लिए अतिरिक्त तरीके से परिचित हैं?
मोदी

1
मेरे पास एक समान आवश्यकता है और कोई एपीस नहीं हैं जो आपको कीस्टोर को जार के हिस्से के रूप में सेट करने दें। @Modi, क्या आप कोई समाधान खोजने में सक्षम थे?
रॉबिन वर्गीज

क्या आप मुख्य स्टोर उपयोग के मामले से मतलब रखते हैं? क्या आप टॉमकैट के साथ स्प्रिंग बूट का उपयोग कर रहे हैं?
मोदी

@RobinVarghese क्या आपको अपनी समस्या का कोई हल मिला। मेरे पास एक समान उपयोग मामला हल किया जाना है। यदि आपके पास कुछ सुझाव है तो सराहना करें।
क्षितिज 7

मामले में अगर tomcat को https का उपयोग करने के लिए keystore लोकेशन की आवश्यकता होती है classpath:filenameतो वह उपयोग कर सकता है ताकि jar के भीतर से keystore फाइल को पढ़ा जा सके।
क्षितिज 7

60

आप स्प्रिंग ढांचे का उपयोग कर रहे हैं तो पढ़ने ClassPathResourceएक में Stringउपयोग करते हुए बहुत सरल है स्प्रिंग ढांचे केFileCopyUtils :

String data = "";
ClassPathResource cpr = new ClassPathResource("static/file.txt");
try {
    byte[] bdata = FileCopyUtils.copyToByteArray(cpr.getInputStream());
    data = new String(bdata, StandardCharsets.UTF_8);
} catch (IOException e) {
    LOG.warn("IOException", e);
}

वहाँ एक बाधा है जहाँ फ़ाइल बैठना चाहिए? क्या यह क्लासपाथ में कहीं भी हो सकता है? क्या यह परियोजना की जड़ में हो सकता है?
स्टीफन

यह क्लासपाथ में कहीं भी हो सकता है।
शुभ

45

यदि आप एक फ़ाइल का उपयोग करना चाहते हैं:

ClassPathResource classPathResource = new ClassPathResource("static/something.txt");

InputStream inputStream = classPathResource.getInputStream();
File somethingFile = File.createTempFile("test", ".txt");
try {
    FileUtils.copyInputStreamToFile(inputStream, somethingFile);
} finally {
    IOUtils.closeQuietly(inputStream);
}

7

जब वसंत बूट परियोजना एक जार के रूप में चल रही है और classpath में कुछ फ़ाइल पढ़ने की आवश्यकता है, तो मैं इसे नीचे दिए गए कोड द्वारा कार्यान्वित करता हूं

Resource resource = new ClassPathResource("data.sql");
BufferedReader reader = new BufferedReader(new InputStreamReader(resource.getInputStream()));
reader.lines().forEach(System.out::println);

3

मैंने एक java 8 तरीके से ClassPathResourceReader क्लास बनाई है ताकि क्लासपाथ से आसानी से पढ़ी जा सकने वाली फाइल बना सकूँ

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.util.stream.Collectors;

import org.springframework.core.io.ClassPathResource;

public final class ClassPathResourceReader {

    private final String path;

    private String content;

    public ClassPathResourceReader(String path) {
        this.path = path;
    }

    public String getContent() {
        if (content == null) {
            try {
                ClassPathResource resource = new ClassPathResource(path);
                BufferedReader reader = new BufferedReader(new InputStreamReader(resource.getInputStream()));
                content = reader.lines().collect(Collectors.joining("\n"));
                reader.close();
            } catch (IOException ex) {
                throw new RuntimeException(ex);
            }
        }
        return content;
    }
}

उपयोग:

String content = new ClassPathResourceReader("data.sql").getContent();

2

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


2
to get list of data from src/main/resources/data folder --
first of all mention your folder location in properties file as - 
resourceLoader.file.location=data

inside class declare your location. 

@Value("${resourceLoader.file.location}")
    @Setter
    private String location;

    private final ResourceLoader resourceLoader;

public void readallfilesfromresources() {
       Resource[] resources;

        try {
            resources = ResourcePatternUtils.getResourcePatternResolver(resourceLoader).getResources("classpath:" + location + "/*.json");
            for (int i = 0; i < resources.length; i++) {
                try {
                InputStream is = resources[i].getInputStream();
                byte[] encoded = IOUtils.toByteArray(is);
                String content = new String(encoded, Charset.forName("UTF-8"));
                }
            }
        } catch (IOException e) {
            throw new UncheckedIOException(e);
        }
}

1

जर्सी को अनपैक जार बनाने की जरूरत है।

<build>  
    <plugins>
        <plugin>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-maven-plugin</artifactId>
            <configuration>
                <requiresUnpack>
                    <dependency>
                        <groupId>com.myapp</groupId>
                        <artifactId>rest-api</artifactId>
                    </dependency>
                </requiresUnpack>
            </configuration>
        </plugin>
    </plugins>
</build>  

0
in spring boot :

1) if your file is ouside jar you can use :        

@Autowired
private ResourceLoader resourceLoader;

**.resource(resourceLoader.getResource("file:/path_to_your_file"))**

2) if your file is inside resources of jar you can `enter code here`use :

**.resource(new ClassPathResource("file_name"))**

0

एंडी के उत्तर के आधार पर मैंने संसाधनों में एक निर्देशिका और उप-निर्देशिकाओं के तहत सभी YAMLs की एक इनपुट स्ट्रीम प्राप्त करने के लिए निम्नलिखित का उपयोग किया (ध्यान दें कि जिस मार्ग से गुजरना शुरू नहीं होता है /):

private static Stream<InputStream> getInputStreamsFromClasspath(
        String path,
        PathMatchingResourcePatternResolver resolver
) {
    try {
        return Arrays.stream(resolver.getResources("/" + path + "/**/*.yaml"))
                .filter(Resource::exists)
                .map(resource -> {
                    try {
                        return resource.getInputStream();
                    } catch (IOException e) {
                        return null;
                    }
                })
                .filter(Objects::nonNull);
    } catch (IOException e) {
        logger.error("Failed to get definitions from directory {}", path, e);
        return Stream.of();
    }
}

0

मैं भी इसी तरह के सिचुएशन में फंस गया, लेकिन बिल्कुल वैसा ही नहीं, संसाधन फ़ोल्डर से एक JSON फाइल पढ़ना चाहता था। ssc / main / Resources इसलिए नीचे एक कोड कुछ इस तरह लिखा है।

स्प्रिंग बूट एप्लिकेशन में फ़ाइल एफ रोम क्लासपाठ को पढ़ने के लिए यहां विभिन्न तरीके दिए गए हैं

@ वाल्यू ("क्लासपैथ: टेस्ट.जसन") निजी संसाधन संसाधन;

File file = resource.getFile();

@Value("classpath:test.json")

निजी संसाधन संसाधन;

File file = resource.getFile();

अब, यह कोड पूरी तरह से ठीक काम करता है एक बार जब मैं इसका उपयोग करता हूं, तो mvan: स्प्रिंग-बूट: रन करता हूं, लेकिन जैसे ही मैं maven का उपयोग करके एप्लिकेशन का निर्माण और पैकेजिंग कर रहा हूं और इसे सरल निष्पादन योग्य जार के रूप में चला रहा हूं, मुझे एक अपवाद मिल रहा है। चलिए अब आगे बढ़ते हैं और इसका समाधान निकालते हैं।

InputStream का उपयोग करके मुझे अपने मामले में उत्तर मिला: -

@Value("classpath:test.json")
    Resource resourceFile;

private void testResource(Resource resource) {
        try {
            InputStream resourcee = resource.getInputStream();
            String text = null;
            try (final Reader reader = new InputStreamReader(resourcee)) {
                text = CharStreams.toString(reader);
            }
            System.out.println(text);

        } catch (IOException e) {
            e.printStackTrace();
        }
    }

वसा जार की अवधारणा पर स्प्रिंग काम करता है, इसलिए यह वास्तव में कठिन है क्योंकि यह ग्रहण में अलग तरह से व्यवहार करता है और जब आप जार के रूप में चलाते हैं।


0

मूल रूप से त्रुटि संदेश के संबंध में

इसे पूर्ण फ़ाइल पथ पर हल नहीं किया जा सकता क्योंकि यह फ़ाइल सिस्टम में नहीं रहता है

पथ समस्या के समाधान के लिए निम्न कोड उपयोगी हो सकता है:

Paths.get("message.txt").toAbsolutePath().toString();

इसके साथ आप यह निर्धारित कर सकते हैं, जहां एप्लिकेशन गुम फ़ाइल की अपेक्षा करता है। आप इसे अपने आवेदन की मुख्य विधि में निष्पादित कर सकते हैं।

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