ฉันจะโหลดไฟล์จากโฟลเดอร์ทรัพยากรได้อย่างไร


240

โครงการของฉันมีโครงสร้างดังต่อไปนี้:

/src/main/java/
/src/main/resources/
/src/test/java/
/src/test/resources/

ฉันมีไฟล์อยู่/src/test/resources/test.csvและฉันต้องการโหลดไฟล์จากการทดสอบหน่วยใน/src/test/java/MyTest.java

ฉันมีรหัสนี้ซึ่งใช้งานไม่ได้ มันบ่นว่า "ไม่มีไฟล์หรือไดเรกทอรีดังกล่าว"

BufferedReader br = new BufferedReader (new FileReader(test.csv))

ฉันก็ลองทำเช่นนี้

InputStream is = (InputStream) MyTest.class.getResourcesAsStream(test.csv))

สิ่งนี้ยังใช้งานไม่ได้ มันกลับnullมา ฉันใช้ Maven เพื่อสร้างโครงการของฉัน


ไม่ทำงานอย่างไร ข้อผิดพลาดของคุณคืออะไร?
Daniel Kaplan

17
ลองสิ่งนี้this.getClass().getResource("/test.csv")
SRy


@ ทำงานได้ดี (เพราะจะทำให้ URL เส้นทางที่แน่นอนกลับมา) แต่เมื่อฉันสร้างไฟล์ jar มันไม่ทำงานเนื่องจากภายใน jar และพา ธ สัมบูรณ์กลายเป็นโมฆะมีวิธีการเล่นกับเส้นทางญาติเอง
ShankPossible

คำตอบ:


240

ลองสิ่งต่อไปนี้:

ClassLoader classloader = Thread.currentThread().getContextClassLoader();
InputStream is = classloader.getResourceAsStream("test.csv");

ถ้าข้างต้นใช้งานไม่ได้โครงการต่าง ๆ ได้ถูกเพิ่มคลาสต่อไปนี้: 1 (รหัสที่นี่ ) 2ClassLoaderUtil

นี่คือตัวอย่างของวิธีการใช้คลาสนั้น:

src \ Main \ Java \ คอม \ บริษัท \ test \ YourCallingClass.java
src \ Main \ Java \ คอม \ opensymphony \ xwork2 \ util \ ClassLoaderUtil.java
src \ Main \ ทรัพยากร \ test.csv
// java.net.URL
URL url = ClassLoaderUtil.getResource("test.csv", YourCallingClass.class);
Path path = Paths.get(url.toURI());
List<String> lines = Files.readAllLines(path, StandardCharsets.UTF_8);
// java.io.InputStream
InputStream inputStream = ClassLoaderUtil.getResourceAsStream("test.csv", YourCallingClass.class);
InputStreamReader streamReader = new InputStreamReader(inputStream, StandardCharsets.UTF_8);
BufferedReader reader = new BufferedReader(streamReader);
for (String line; (line = reader.readLine()) != null;) {
    // Process line
}

หมายเหตุ

  1. เห็นมันในWayback เครื่อง
  2. นอกจากนี้ในGitHub

14
ขอบคุณสำหรับคำตอบคุณช่วยอธิบายได้ไหมว่าทำไมเราควรใช้ตัวโหลดที่เจาะจงมาก ๆ แต่ไม่ใช่ของคลาส?
Hui Wang

1
ยอดเยี่ยมฉันโง่มากที่ฉันใช้Object.class.getClassLoader();จากบริบทแบบคงที่ซึ่งไม่ได้ผลข้อเสนอแนะนี้ทำได้ดีเกือบจะ%20เป็นช่องว่างที่ให้ฉันด้วยFileNotFoundException
ycomp

5
@ycomp เนื่องจาก getResource ส่งคืน URL ไม่ใช่ไฟล์ เมธอด getFile ของ java.net.URL ไม่แปลง URL เป็นไฟล์ มันเพิ่งส่งกลับเส้นทางและส่วนแบบสอบถามของ URL คุณไม่ควรลองแปลงเป็นไฟล์ เพียงแค่เรียก openStream และอ่านจากที่
VGR

ตรวจสอบโครงการนี้แก้ไขการสแกนโฟลเดอร์ทรัพยากร: github.com/fraballi/resources-folder-scanner
Felix Aballi

60

ลอง:

InputStream is = MyTest.class.getResourceAsStream("/test.csv");

IIRC getResourceAsStream()โดยค่าเริ่มต้นสัมพันธ์กับแพ็คเกจของคลาส

ตามที่ @Terran ตั้งข้อสังเกตอย่าลืมที่จะเพิ่ม/ที่จุดเริ่มต้นของชื่อไฟล์


3
สำหรับผู้ที่กำลังมองหาการโพสต์นี้นำคุณไปสู่การได้รับทรัพยากรในฐานะString stackoverflow.com/a/35446009/544045
เทรเวอร์

12
"/" เป็นกุญแจสำคัญ
Terran

33

นี่คือวิธีแก้ปัญหาอย่างรวดเร็วด้วยการใช้Guava :

import com.google.common.base.Charsets;
import com.google.common.io.Resources;

public String readResource(final String fileName, Charset charset) throws IOException {
        return Resources.toString(Resources.getResource(fileName), charset);
}

การใช้งาน:

String fixture = this.readResource("filename.txt", Charsets.UTF_8)

28

ลองใช้รหัส Flow ในโครงการ Spring

ClassPathResource resource = new ClassPathResource("fileName");
InputStream inputStream = resource.getInputStream();

หรือในโครงการที่ไม่ใช่สปริง

 ClassLoader classLoader = getClass().getClassLoader();
 File file = new File(classLoader.getResource("fileName").getFile());
 InputStream inputStream = new FileInputStream(file);

ควรปิด InputStream หรือไม่?
030

7

โครงการที่ไม่ใช่สปริง:

String filePath = Objects.requireNonNull(MyClass.class.getClassLoader().getResource("any.json")).getPath();

Stream<String> lines = Files.lines(Paths.get(filePath));

สำหรับโครงการสปริงคุณสามารถใช้รหัสบรรทัดเดียวเพื่อรับไฟล์ใด ๆ ภายใต้โฟลเดอร์ทรัพยากร:

File file = ResourceUtils.getFile(ResourceUtils.CLASSPATH_URL_PREFIX + "any.json");

String content = new String(Files.readAllBytes(file.toPath()));

5

ตอนนี้ฉันกำลังอธิบายซอร์สโค้ดสำหรับการอ่านฟอนต์จาก maven ไดเรกทอรีทรัพยากรที่สร้างขึ้น

SCR / หลัก / ทรัพยากร / calibril.ttf

ป้อนคำอธิบายรูปภาพที่นี่

Font getCalibriLightFont(int fontSize){
    Font font = null;
    try{
        URL fontURL = OneMethod.class.getResource("/calibril.ttf");
        InputStream fontStream = fontURL.openStream();
        font = new Font(Font.createFont(Font.TRUETYPE_FONT, fontStream).getFamily(), Font.PLAIN, fontSize);
        fontStream.close();
    }catch(IOException | FontFormatException ief){
        font = new Font("Arial", Font.PLAIN, fontSize);
        ief.printStackTrace();
    }   
    return font;
}

มันใช้งานได้สำหรับฉันและหวังว่าซอร์สโค้ดทั้งหมดจะช่วยให้คุณสนุกได้เลย!


5

สำหรับ java หลังจาก 1.7

 List<String> lines = Files.readAllLines(Paths.get(getClass().getResource("test.csv").toURI()));

ฉันต้องใช้ "/test.csv" (หมายเหตุเครื่องหมายทับ)
Leon

4
ClassLoader loader = Thread.currentThread().getContextClassLoader();
InputStream is = loader.getResourceAsStream("test.csv");

ถ้าคุณใช้บริบท ClassLoader เพื่อค้นหาทรัพยากรแน่นอนว่ามันจะเสียค่าใช้จ่ายประสิทธิภาพของแอปพลิเคชัน


4
โปรแกรมเมอร์ควรกังวลเกี่ยวกับประสิทธิภาพเสมอ ในขณะที่หลีกเลี่ยงการปรับให้เหมาะสมก่อนวัยอันควรอย่างแน่นอนการรู้วิธีการที่มีประสิทธิภาพมากขึ้นนั้นดีเสมอ มันเหมือนกับการรู้ความแตกต่างระหว่าง LinkedList และ ArrayList และเมื่อใช้อย่างใดอย่างหนึ่ง
Marvo

6
@Marvo: โปรแกรมเมอร์จะต้องกังวลเกี่ยวกับคุณภาพความสามารถและความสะดวกในการบำรุงรักษาประสิทธิภาพอยู่ที่คิว
Igor Rodriguez

2

นำเข้าสิ่งต่อไปนี้:

import java.io.IOException;
import java.io.FileNotFoundException;
import java.io.BufferedReader;
import java.io.InputStreamReader;
import java.io.InputStream;
import java.util.ArrayList;

วิธีการต่อไปนี้จะส่งคืนไฟล์ใน ArrayList of Strings:

public ArrayList<String> loadFile(String filename){

  ArrayList<String> lines = new ArrayList<String>();

  try{

    ClassLoader classloader = Thread.currentThread().getContextClassLoader();
    InputStream inputStream = classloader.getResourceAsStream(filename);
    InputStreamReader streamReader = new InputStreamReader(inputStream, StandardCharsets.UTF_8);
    BufferedReader reader = new BufferedReader(streamReader);
    for (String line; (line = reader.readLine()) != null;) {
      lines.add(line);
    }

  }catch(FileNotFoundException fnfe){
    // process errors
  }catch(IOException ioe){
    // process errors
  }
  return lines;
}

2

ผมประสบปัญหาเดียวกัน

ไม่พบไฟล์โดยตัวโหลดคลาสซึ่งหมายความว่าไม่ได้บรรจุลงในส่วน (jar) คุณจำเป็นต้องสร้างโครงการ ตัวอย่างเช่นด้วย maven:

mvn clean install

ดังนั้นไฟล์ที่คุณเพิ่มลงในโฟลเดอร์ทรัพยากรจะเข้าสู่ maven build และพร้อมใช้งานกับแอปพลิเคชัน

ผมอยากจะให้คำตอบของฉันมันไม่ได้อธิบายวิธีการอ่านแฟ้ม (คำตอบอื่น ๆ ทำอธิบายว่า) มันตอบว่าทำไม InputStreamหรือresourceเป็นโมฆะ ที่คล้ายกันคำตอบอยู่ที่นี่


1
ขอบคุณช่วยฉันด้วยการคิดว่าฉันทำใจไม่อยู่!
StuPointerException

1

getResource () ทำงานได้ดีกับไฟล์ทรัพยากรที่วางไว้src/main/resourcesเท่านั้น ในการรับไฟล์ที่อยู่ในพา ธ นอกเหนือจากที่src/main/resourcesบอกว่าsrc/test/javaคุณจำเป็นต้องสร้างไฟล์ขึ้นมาอย่างประณีต

ตัวอย่างต่อไปนี้อาจช่วยคุณได้

import java.io.BufferedReader;
import java.io.FileReader;
import java.io.IOException;
import java.net.URISyntaxException;
import java.net.URL;

public class Main {
    public static void main(String[] args) throws URISyntaxException, IOException {
        URL location = Main.class.getProtectionDomain().getCodeSource().getLocation();
        BufferedReader br = new BufferedReader(new FileReader(location.getPath().toString().replace("/target/classes/", "/src/test/java/youfilename.txt")));
    }
}

0

รหัสทำงานเมื่อไม่ได้ใช้ Maven-build jar ตัวอย่างเช่นเมื่อรันจาก IDE ของคุณหรือไม่ ถ้าเป็นเช่นนั้นตรวจสอบให้แน่ใจว่าไฟล์นั้นมีอยู่ในขวดจริง ทรัพยากรโฟลเดอร์ควรจะรวมอยู่ในไฟล์ pom <build><resources>ใน


เมื่อใช้ eclipse และใช้รหัสจาก IDE เอง เราจะโหลดทรัพยากรที่ "/ src / test / resources" ในโค้ด Java โดยเฉพาะในการทดสอบหน่วยได้อย่างไร พิจารณาโครงสร้างโครงการ maven มาตรฐาน
Bhavesh

0

ชั้นต่อไปนี้สามารถนำมาใช้ในการโหลดresourceจากและยังได้รับข้อความแสดงข้อผิดพลาดที่เหมาะสมในกรณีที่มีปัญหาเกี่ยวกับการที่ได้รับclasspathfilePath

import java.io.InputStream;
import java.nio.file.NoSuchFileException;

public class ResourceLoader
{
    private String filePath;

    public ResourceLoader(String filePath)
    {
        this.filePath = filePath;

        if(filePath.startsWith("/"))
        {
            throw new IllegalArgumentException("Relative paths may not have a leading slash!");
        }
    }

    public InputStream getResource() throws NoSuchFileException
    {
        ClassLoader classLoader = this.getClass().getClassLoader();

        InputStream inputStream = classLoader.getResourceAsStream(filePath);

        if(inputStream == null)
        {
            throw new NoSuchFileException("Resource file not found. Note that the current directory is the source folder!");
        }

        return inputStream;
    }
}

0

ฉันแก้ไขปัญหานี้ด้วยการเพิ่ม/scr/main/resourcesโฟลเดอร์ในของฉันJava Build Path

ป้อนคำอธิบายรูปภาพที่นี่


ลองสิ่งนี้ แต่นี่ไม่ใช่วิธีแก้ปัญหา
Jasonw

0

ฉันทำให้มันทำงานได้ทั้งบน jar และ IDE โดยการเขียนเป็น

 InputStream schemaStream = ProductUtil.class.getClassLoader().getResourceAsStream(jsonSchemaPath);
            byte[] buffer = new byte[schemaStream.available()];
            schemaStream.read(buffer);

        File tempFile = File.createTempFile("com/package/schema/testSchema", "json");
        tempFile.deleteOnExit();
        FileOutputStream out = new FileOutputStream(tempFile);
        out.write(buffer);

โครงสร้างไฟล์ของคุณมีลักษณะอย่างไร
luckydonald


0

คุณสามารถใช้ com.google.common.io.Resources.getResource เพื่ออ่าน URL ของไฟล์จากนั้นรับเนื้อหาไฟล์โดยใช้ java.nio.file.Files เพื่ออ่านเนื้อหาของไฟล์

URL urlPath = Resources.getResource("src/main/resource");
List<String> multilineContent= Files.readAllLines(Paths.get(urlPath.toURI()));

-2

ฉันทำให้มันทำงานโดยไม่มีการอ้างอิงถึง "class" หรือ "ClassLoader"

สมมติว่าเรามีสถานการณ์จำลองสามสถานการณ์ด้วยที่ตั้งของไฟล์ 'example.file' และไดเรกทอรีทำงานของคุณ (ที่ที่แอปของคุณดำเนินการ) คือ home / mydocuments / program / projects / myapp:

ก) การสืบทอดโฟลเดอร์ย่อยไปยังไดเร็กทอรีการทำงาน: myapp / res / files / example.file

b) โฟลเดอร์ย่อยที่ไม่สืบทอดในไดเรกทอรีการทำงาน: projects / files / example.file

b2) โฟลเดอร์ย่อยอื่นที่ไม่สืบทอดมาสู่ไดเรกทอรีการทำงาน: program / files / example.file

c) โฟลเดอร์รูท: home / mydocuments / files / example.file (Linux; ใน Windows แทนที่ home / ด้วย C :)

1) หาเส้นทางที่ถูกต้อง: a) String path = "res/files/example.file"; b) String path = "../projects/files/example.file" b2) String path = "../../program/files/example.file" c)String path = "/home/mydocuments/files/example.file"

โดยทั่วไปถ้าเป็นโฟลเดอร์รูทให้เริ่มต้นชื่อพา ธ ด้วยเครื่องหมายสแลช หากเป็นโฟลเดอร์ย่อยต้องไม่มีเครื่องหมายทับอยู่หน้าชื่อพา ธ หากโฟลเดอร์ย่อยไม่ได้อยู่ในไดเรกทอรีที่ใช้งานได้คุณต้องใช้ซีดีโดยใช้ "../" สิ่งนี้บอกให้ระบบเพิ่มขึ้นหนึ่งโฟลเดอร์

2) สร้างวัตถุไฟล์โดยผ่านเส้นทางที่ถูกต้อง:

File file = new File(path);

3) ตอนนี้คุณพร้อมแล้วที่จะไป:

BufferedReader br = new BufferedReader(new FileReader(file));
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.