web-dev-qa-db-ja.com

jar内のファイルをjarの外にコピーする方法は?

Jarからファイルをコピーしたい。コピーするファイルは、作業ディレクトリの外部にコピーされます。私はいくつかのテストを行い、すべてのメソッドを試してみましたが、結果は0バイトのファイルになりました。

[〜#〜] edit [〜#〜]:ファイルのコピーを手動ではなくプログラムを介して行いたい。

40
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 で回答し、 Andy Thomas で指摘したように、次の方法でこれを達成できます。

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

詳細については、 GOXR3PLUS answer を参照してください

49
Ordiel

0バイトのファイルについてのあなたのコメントを考えると、あなたはプログラムでこれをやろうとしていることを前提にしなければなりません。それが当てはまる場合は、 Class.getResource() を使用してJAR内のファイルを指すURLを取得し、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);

ほとんどの場合、現在のコードの問題は、バッファリングされた出力ストリームを(正しく)使用してファイルに書き込みますが、(誤って)閉じられないことです。

ああ、質問を編集して、正確にhowを明確にする必要があります(プログラムではなく、言語で...)

34
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ファイルを解凍しようとしました

12
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で現在の作業ディレクトリを取得する

11
GOXR3PLUS

堅牢なソリューション:

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());
1
zoran

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

Jarから外部にファイルをコピーするには、次のアプローチを使用する必要があります。

  1. getResourceAsStream() を使用して、jarファイル内のファイルにInputStreamを取得します
  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