Як завантажити файл із папки ресурсів?


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 для створення свого проекту.


Як не працює? Яка ваша помилка?
Даніель Каплан

17
спробуйте цеthis.getClass().getResource("/test.csv")
SRy


@SRy це спрацьовувало (тому що це дасть абсолютний URL-адресу шляху у відповідь), але в момент, коли я створю файл jar, він не працює, оскільки його всередині банку і абсолютний шлях стає недійсним, чи є спосіб грати з самим відносним шляхом
ShankPossible

Відповіді:


240

Спробуйте наступне:

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

Якщо вищезгадане не працює, додаються різні проекти наступного класу: 1 (код тут ). 2ClassLoaderUtil

Ось кілька прикладів використання цього класу:

src \ main \ java \ com \ company \ test \ YourCallingClass.java
src \ main \ java \ com \ 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. Дивіться це в машині The Wayback .
  2. Також у GitHub .

14
THX для відповіді, чи можете ви пояснити, чому ми повинні використовувати цей дуже специфічний навантажувач, але не той, що класу?
Хуй Ван

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
Фелікс Абаллі

60

Спробуйте:

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

IIRC getResourceAsStream()за замовчуванням є відносно пакету класу.

Як зазначав @Terran, не забудьте додати ім'я /на початку файлу


3
Для тих, хто шукає розуму, ця публікація пропонує вам весь шлях до отримання ресурсу як String stackoverflow.com/a/35446009/544045
Тревор

12
Ключ "/" є.
Терран

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

Спробуйте поточні коди проекту 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 / main / ресурси / 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" (зверніть увагу на косу рису).
Леон

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

Якщо ви використовуєте контекст ClassLoader для пошуку ресурсу, то, безумовно, це буде коштувати продуктивності програми.


4
Програмісти завжди повинні дбати про продуктивність. Хоча передчасної оптимізації, безумовно, слід уникати, знаючи, що ефективніший підхід застосовувати завжди добре. Це як би знати різницю між LinkedList і ArrayList і коли використовувати те чи інше.
Марво

6
@Marvo: Програмісти завжди повинні піклуватися про якість, можливості та простоту обслуговування, продуктивність на черзі.
Ігор Родрігес

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

Я зіткнувся з тим же питанням .

Завантажувач класу не знайшов файл, а це означає, що він не був запакований у артефакт (банку). Вам потрібно побудувати проект . Наприклад, з 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

Чи працює код, коли не працює jar Maven-build, наприклад, при запуску з IDE? Якщо так, переконайтеся, що файл дійсно включений у банку. Папка ресурсів повинна бути включена у файл pom, в <build><resources>.


При використанні затемнення та запуску коду від самої IDE. Як ми можемо завантажувати ресурси, що знаходяться на "/ src / test / resources" в коді Java, зокрема в одиничних тестах. Розглянемо стандартну структуру проекту Maven.
Бхавеш

0

Наступний клас може бути використаний для завантаження a resourceз, classpathа також отримання повідомлення про помилку примірника у випадку, якщо є проблема із заданим filePath.

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

Я змусив його працювати як на запущеній банці, так і на 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:

a) Підпапка, що має потомство до робочого каталогу: myapp / res / files / example.file

b) Підпапка, не нащадка до робочого каталогу: Проекти / файли / example.file

b2) Ще одна підпапка, яка не є нащадком робочого каталогу: program / files / example.file

c) Коренева папка: home / mydocuments / files / example.file (Linux; в Windows замінити home / на C :)

1) Отримайте правильний шлях: а) String path = "res/files/example.file"; б) String path = "../projects/files/example.file" b2) String path = "../../program/files/example.file" в)String path = "/home/mydocuments/files/example.file"

В основному, якщо це коренева папка, почніть назву шляху з провідним косою косою. Якщо це підпапка, жодна коса риса не повинна бути перед назвою шляху. Якщо підпапка не є нащадком робочої директорії, ви повинні створити її, використовуючи "../". Це повідомляє системі перейти на одну папку.

2) Створіть об'єкт File, пройшовши правильний шлях:

File file = new File(path);

3) Ви зараз готові йти:

BufferedReader br = new BufferedReader(new FileReader(file));
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.