web-dev-qa-db-ja.com

jarファイル内にパッケージ化されているexeを実行します

Javaプログラムを介してexeを実行しています。パスはJavaでハードコードされています。

Exeファイルをjarファイルにパッケージ化しました。

しかし、パス名がJavaファイルにハードコードされているため、スタックしているため、jarをスタンドアロンプ​​ログラムとして実行できません。

そのようなjarをパッケージ化するためのヒントはありますか?つまり、内部にexeがあり、スタンドアロンプ​​ログラムとして実行できますか?

25
krisp

これにより、.exeがローカルディスク上のローカルファイルに抽出されます。 Javaプログラムが存在する場合、ファイルは削除されます。

import Java.io.Closeable;
import Java.io.File;
import Java.io.FileNotFoundException;
import Java.io.FileOutputStream;
import Java.io.IOException;
import Java.io.InputStream;
import Java.io.OutputStream;
import Java.net.URI;
import Java.net.URISyntaxException;
import Java.net.URL;
import Java.security.CodeSource;
import Java.security.ProtectionDomain;
import Java.util.Zip.ZipEntry;
import Java.util.Zip.ZipException;
import Java.util.Zip.ZipFile;

public class Main
{
    public static void main(final String[] args)
        throws URISyntaxException,
               ZipException,
               IOException
    {
        final URI uri;
        final URI exe;

        uri = getJarURI();
        exe = getFile(uri, "Main.class");
        System.out.println(exe);
    }

    private static URI getJarURI()
        throws URISyntaxException
    {
        final ProtectionDomain domain;
        final CodeSource       source;
        final URL              url;
        final URI              uri;

        domain = Main.class.getProtectionDomain();
        source = domain.getCodeSource();
        url    = source.getLocation();
        uri    = url.toURI();

        return (uri);
    }

    private static URI getFile(final URI    where,
                               final String fileName)
        throws ZipException,
               IOException
    {
        final File location;
        final URI  fileURI;

        location = new File(where);

        // not in a JAR, just return the path on disk
        if(location.isDirectory())
        {
            fileURI = URI.create(where.toString() + fileName);
        }
        else
        {
            final ZipFile zipFile;

            zipFile = new ZipFile(location);

            try
            {
                fileURI = extract(zipFile, fileName);
            }
            finally
            {
                zipFile.close();
            }
        }

        return (fileURI);
    }

    private static URI extract(final ZipFile zipFile,
                               final String  fileName)
        throws IOException
    {
        final File         tempFile;
        final ZipEntry     entry;
        final InputStream  zipStream;
        OutputStream       fileStream;

        tempFile = File.createTempFile(fileName, Long.toString(System.currentTimeMillis()));
        tempFile.deleteOnExit();
        entry    = zipFile.getEntry(fileName);

        if(entry == null)
        {
            throw new FileNotFoundException("cannot find file: " + fileName + " in archive: " + zipFile.getName());
        }

        zipStream  = zipFile.getInputStream(entry);
        fileStream = null;

        try
        {
            final byte[] buf;
            int          i;

            fileStream = new FileOutputStream(tempFile);
            buf        = new byte[1024];
            i          = 0;

            while((i = zipStream.read(buf)) != -1)
            {
                fileStream.write(buf, 0, i);
            }
        }
        finally
        {
            close(zipStream);
            close(fileStream);
        }

        return (tempFile.toURI());
    }

    private static void close(final Closeable stream)
    {
        if(stream != null)
        {
            try
            {
                stream.close();
            }
            catch(final IOException ex)
            {
                ex.printStackTrace();
            }
        }
    }
}
34
TofuBeer

オペレーティングシステムは.jarファイルを気にしないか認識していないため、実行する前に.exeファイルを一時的な場所に解凍する必要があります。

11
Joachim Sauer
//gets program.exe from inside the JAR file as an input stream
InputStream is = getClass().getResource("program.exe").openStream();
//sets the output stream to a system folder
OutputStream os = new FileOutputStream("program.exe");

//2048 here is just my preference
byte[] b = new byte[2048];
int length;

while ((length = is.read(b)) != -1) {
    os.write(b, 0, length);
}

is.close();
os.close();
5
D.Snap

他のユーザーが質問に正しく回答している間に、抽出して実行し、クリーンアップします。考慮すべきもう1つのポイントは、完全にネイティブになることです。

特定のタスクを実行するために、すでにネイティブバイナリを使用しています。また、アプリケーションをインストールするネイティブインストーラーを作成し、バイナリをOS固有の場所(Win32のProgram Files)にインストールして、適切なショートカットを作成してみませんか。

このようにすると、アプリケーションはよりネイティブに感じられ、この事実を回避するためにコードを記述したり管理したりする必要がなくなります。現時点では、Jarはクロスプラットフォームのコードのように見えますが(Jarはどこでも実行できますか?)、どこでも実行されないネイティブバイナリをパックしています。これは矛盾のように感じます。

インストーラーには、学ぶべき優れたチュートリアルとコードサンプルがたくさんあるので、Nullsoft Installation System(NSIS)をお勧めします。

3

使用する

getClass().getResource(what).openStream()

ディスク内の別のファイルにコピーします。

2
Mid Bifroid