it-roy-ru.com

Как скопировать файл внутри баночки наружу?

Я хочу скопировать файл из банки. Файл, который я копирую, будет скопирован за пределы рабочего каталога. Я сделал несколько тестов и все методы, которые я пытаюсь получить, заканчиваются 0-байтовыми файлами.

EDIT: Я хочу, чтобы копирование файла осуществлялось с помощью программы, а не вручную.

38
jtl999

Прежде всего я хочу сказать, что некоторые ответы, опубликованные ранее, являются полностью правильными, но я хочу дать свои, поскольку иногда мы не можем использовать библиотеки с открытым исходным кодом под лицензией GPL или потому, что нам лень скачивать jar XD или что-то еще. когда-либо ваша причина здесь является автономным решением.

Функция ниже копирует ресурс рядом с файлом Jar:

  /**
     * Export a resource embedded into a Jar file to the local file path.
     *
     * @param resourceName ie.: "/SmartLibrary.dll"
     * @return The path to the exported resource
     * @throws Exception
     */
    static public String ExportResource(String resourceName) throws Exception {
        InputStream stream = null;
        OutputStream resStreamOut = null;
        String jarFolder;
        try {
            stream = ExecutingClass.class.getResourceAsStream(resourceName);//note that each / is a directory down in the "jar tree" been the jar the root of the tree
            if(stream == null) {
                throw new Exception("Cannot get resource \"" + resourceName + "\" from Jar file.");
            }

            int readBytes;
            byte[] buffer = new byte[4096];
            jarFolder = new File(ExecutingClass.class.getProtectionDomain().getCodeSource().getLocation().toURI().getPath()).getParentFile().getPath().replace('\\', '/');
            resStreamOut = new FileOutputStream(jarFolder + resourceName);
            while ((readBytes = stream.read(buffer)) > 0) {
                resStreamOut.write(buffer, 0, readBytes);
            }
        } catch (Exception ex) {
            throw ex;
        } finally {
            stream.close();
            resStreamOut.close();
        }

        return jarFolder + resourceName;
    }

Просто измените ExecutingClass на имя вашего класса и назовите его так:

String fullPath = ExportResource("/myresource.ext");

Редактировать для Java 7+ (для вашего удобства)

Как ответил GOXR3PLUS и отметил Энди Томас вы можете добиться этого с помощью:

Files.copy( InputStream in, Path target, CopyOption... options)

Смотрите GOXR3PLUS ответ для более подробной информации

47
Ordiel

Учитывая ваш комментарий о 0-байтовых файлах, я должен предположить, что вы пытаетесь сделать это программно, и, учитывая ваши теги, вы делаете это в Java. Если это так, тогда просто используйте Class.getResource () , чтобы получить URL, указывающий на файл в вашем JAR, а затем Apache Commons IO FileUtils.copyURLToFile () , чтобы скопировать его в файловая система. Например.:

URL inputUrl = getClass().getResource("/absolute/path/of/source/in/jar/file");
File dest = new File("/path/to/destination/file");
FileUtils.copyURLToFile(inputUrl, dest);

Скорее всего, проблема с любым имеющимся у вас кодом состоит в том, что вы (правильно) используете буферизованный поток вывода для записи в файл, но (неправильно) не можете его закрыть.

О, и вы должны отредактировать свой вопрос, чтобы уточнить как вы хотите это сделать (программно, а не языком, ...)

33
Ryan Stewart

Java 8 (на самом деле FileSystem существует с 1.7) поставляется с некоторыми классными новыми классами/методами для решения этой проблемы. Как кто-то уже упоминал, что JAR - это в основном Zip-файл, вы можете использовать

final URI jarFileUril = URI.create("jar:file:" + file.toURI().getPath());
final FileSystem fs = FileSystems.newFileSystem(jarFileUri, env);

(Смотрите Zip файл )

Тогда вы можете использовать один из удобных методов, таких как:

fs.getPath("filename");

Тогда вы можете использовать класс Files

try (final Stream<Path> sources = Files.walk(from)) {
     sources.forEach(src -> {
         final Path dest = to.resolve(from.relativize(src).toString());
         try {
            if (Files.isDirectory(from)) {
               if (Files.notExists(to)) {
                   log.trace("Creating directory {}", to);
                   Files.createDirectories(to);
               }
            } else {
                log.trace("Extracting file {} to {}", from, to);
                Files.copy(from, to, StandardCopyOption.REPLACE_EXISTING);
            }
       } catch (IOException e) {
           throw new RuntimeException("Failed to unzip file.", e);
       }
     });
}

Примечание: я пытался распаковать JAR-файлы для тестирования

11
Lukino

Более быстрый способ сделать это с помощью Java 7+, плюс код для получения текущего каталога:

   /**
     * Copy a file from source to destination.
     *
     * @param source
     *        the source
     * @param destination
     *        the destination
     * @return True if succeeded , False if not
     */
    public static boolean copy(InputStream source , String destination) {
        boolean succeess = true;

        System.out.println("Copying ->" + source + "\n\tto ->" + destination);

        try {
            Files.copy(source, Paths.get(destination), StandardCopyOption.REPLACE_EXISTING);
        } catch (IOException ex) {
            logger.log(Level.WARNING, "", ex);
            succeess = false;
        }

        return succeess;

    }

Тестирование (icon.png - это изображение внутри образа пакета приложения):

copy(getClass().getResourceAsStream("/image/icon.png"),getBasePathForClass(Main.class)+"icon.png");

О строке кода (getBasePathForClass(Main.class)): -> проверьте ответ, который я добавил здесь :) -> Получение текущего рабочего каталога в Java

9
GOXR3PLUS

Используйте JarInputStream класс:

// assuming you already have an InputStream to the jar file..
JarInputStream jis = new JarInputStream( is );
// get the first entry
JarEntry entry = jis.getNextEntry();
// we will loop through all the entries in the jar file
while ( entry != null ) {
  // test the entry.getName() against whatever you are looking for, etc
  if ( matches ) {
    // read from the JarInputStream until the read method returns -1
    // ...
    // do what ever you want with the read output
    // ...
    // if you only care about one file, break here 
  }
  // get the next entry
  entry = jis.getNextEntry();
}
jis.close();

Смотрите также: JarEntry

1
jarrad

Надежное решение:

public static void copyResource(String res, String dest, Class c) throws IOException {
    InputStream src = c.getResourceAsStream(res);
    Files.copy(src, Paths.get(dest), StandardCopyOption.REPLACE_EXISTING);
}

Вы можете использовать это так:

File tempFileGdalZip = File.createTempFile("temp_gdal", ".Zip");
copyResource("/gdal.Zip", tempFileGdalZip.getAbsolutePath(), this.getClass());
0
zoran

Чтобы скопировать файл из вашего jar-файла, вы должны использовать следующий подход:

  1. Получите InputStream в файл внутри вашего jar-файла, используя getResourceAsStream()
  2. Мы открываем наш целевой файл с помощью FileOutputStream
  3. Копируем байты из входного потока в выходной
  4. Мы закрываем наши потоки, чтобы предотвратить утечку ресурсов

Пример кода, который также содержит переменную, которая не заменяет существующие значения:

public File saveResource(String name) throws IOException {
    return saveResource(name, true);
}

public File saveResource(String name, boolean replace) throws IOException {
    return saveResource(new File("."), name, replace)
}

public File saveResource(File outputDirectory, String name) throws IOException {
    return saveResource(outputDirectory, name, true);
}

public File saveResource(File outputDirectory, String name, boolean replace)
       throws IOException {
    File out = new File(outputDirectory, name);
    if (!replace && out.exists()) 
        return out;
    // Step 1:
    InputStream resource = this.getClass().getResourceAsStream(name);
    if (resource == null)
       throw new FileNotFoundException(name + " (resource not found)");
    // Step 2 and automatic step 4
    try(InputStream in = resource;
        OutputStream writer = new BufferedOutputStream(
            new FileOutputStream(out))) {
         // Step 3
         byte[] buffer = new byte[1024 * 4];
         int length;
         while((length = in.read(buffer)) >= 0) {
             writer.write(buffer, 0, length);
         }
     }
     return out;
}
0
Ferrybig