Как загрузить файл из папки ресурсов?

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

codereviewanskquestions
источник
Не работает как? В чем ваша ошибка?
Даниэль Каплан
17
попробуйте этоthis.getClass().getResource("/test.csv")
SRy
@ SRy это сработало (потому что это даст абсолютный URL-адрес пути взамен), однако в тот момент, когда я создаю jar-файл, он не работает, так как его внутри jar и абсолютный путь становится недействительным, есть ли способ поиграть с самим относительным путем
ShankPossible

Ответы:

240

Попробуйте следующее:

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

Если вышеупомянутое не работает, к различным проектам был добавлен следующий класс: 1 (код здесь ). 2ClassLoaderUtil

Вот несколько примеров использования этого класса:

ЦСИ \ главная \ Java \ ком \ компания \ тест \ YourCallingClass.java
ЦСИ \ главная \ Java \ COM \ OpenSymphony \ xwork2 \ Util \ ClassLoaderUtil.java
ЦСИ \ основные \ ресурсы \ 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 Machine .
  2. Также в GitHub .
Пол Варгас
источник
14
спасибо за ответ, не могли бы вы объяснить, почему мы должны использовать этот очень специфический загрузчик, а не класс?
Хуэй Ван
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, не забудьте добавить /в начале имени файла

vanza
источник
3
Для тех, кто ищет здравомыслие, этот пост поможет вам получить ресурс в виде String stackoverflow.com/a/35446009/544045
Тревор,
12
«/» Является ключом.
Терран
33

Вот одно быстрое решение с использованием гуавы :

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)
Datageek
источник
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);
srsajid
источник
Разве InputStream не должен быть закрыт?
0:30
7

Не весенний проект:

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

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

Для проектов Spring вы также можете использовать однострочный код, чтобы получить любой файл в папке ресурсов:

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

String content = new String(Files.readAllBytes(file.toPath()));
Fuat
источник
5

Теперь я иллюстрирую исходный код для чтения шрифта из каталога созданных Maven ресурсов,

экр / главная / ресурсы / 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;
}

Это сработало для меня и надеюсь, что весь исходный код также поможет вам, Наслаждайтесь!

ArifMustafa
источник
5

Для Java после 1,7

 List<String> lines = Files.readAllLines(Paths.get(getClass().getResource("test.csv").toURI()));
phray2002
источник
Я должен использовать "/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;
}
shalamus
источник
2

Я столкнулся с той же проблемой .

Файл не был найден загрузчиком классов, что означает, что он не был упакован в артефакт (jar). Вам нужно построить проект . Например, с Maven:

mvn clean install

Таким образом, файлы, которые вы добавили в папку ресурсов, попадут в сборку maven и станут доступны для приложения.

Я хотел бы сохранить свой ответ: он не объясняет, как читать файл (другие ответы объясняют это), он отвечает, почему 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, например, при запуске из вашей IDE? Если это так, убедитесь, что файл действительно включен в банку. Папка ресурсов должна быть включена в файл pom, в <build><resources>.

Йорн
источник
При использовании eclipse и запуске кода из самой IDE. Как мы можем загрузить ресурсы, расположенные в "/ src / test / resources" в коде Java, особенно в модульных тестах. Рассмотрим стандартную структуру проекта Maven.
Бхавеш
0

Следующий класс может быть использован для загрузки 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;
    }
}
BullyWiiPlaza
источник
0

Я решил эту проблему, добавив /scr/main/resourcesпапку в мойJava Build Path

введите описание изображения здесь

sklimkovitch
источник
попробуйте это, но это не решение проблемы
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);
rakeeee
источник
Как выглядит ваша файловая структура?
luckydonald
0
this.getClass().getClassLoader().getResource("filename").getPath()
Шива Кумар
источник
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

Я получаю его работать без каких-либо ссылок на «класс» или «ClassLoader».

Допустим, у нас есть три сценария с расположением файла example.file и вашим рабочим каталогом (где выполняется ваше приложение) home / mydocuments / program / projects / myapp:

a) Потомок подпапки к рабочему каталогу: myapp / res / files / example.file

б) подпапка, не являющаяся потомком рабочего каталога: projects / files / 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" б2) 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));
AndreGraveler
источник