21

可能重复:
在 jar 文件中执行 shell 脚本。如何提取?

我在“引擎”包中的 JAR 文件中打包了一个 shell 脚本。

在我的程序中,我正在使用 Process 和 ProcessBuilder 运行一个 shell 命令。这一切都很好。

如果我在我的计算机上指定 shell 脚本的路径,那么程序可以正常工作。但是,如果我将 shell 脚本打包到我的 JAR 中并像这样访问它:

scriptLocation = this.getClass().getResource("/engine/shell-script.sh").toString();

并运行程序,然后我得到以下错误:

java.io.IOException: Cannot run program "file:/Users/USERNAME/Desktop/Application-Name.jar!/engine/shell-script.sh": error=2, No such file or directory
    at java.lang.ProcessBuilder.start(ProcessBuilder.java:460)
    at java.lang.Runtime.exec(Runtime.java:593)
    at java.lang.Runtime.exec(Runtime.java:431)
    at java.lang.Runtime.exec(Runtime.java:328)
    at engine.Download$1.run(Download.java:39)
    at java.lang.Thread.run(Thread.java:680)
Caused by: java.io.IOException: error=2, No such file or directory
    at java.lang.UNIXProcess.forkAndExec(Native Method)
    at java.lang.UNIXProcess.<init>(UNIXProcess.java:53)
    at java.lang.ProcessImpl.start(ProcessImpl.java:91)
    at java.lang.ProcessBuilder.start(ProcessBuilder.java:453)
    ... 5 more

我开始认为这是无法做到的,我非常想提供这个 shell 脚本作为应用程序的一部分,可以做到吗?

提前谢谢大家。

======== 更新 ==========

我最终解决了这个问题。Shell 不能在 JAR 包中执行脚本,它不是关于 Java,而是关于 shell。我通过执行以下操作解决了这个问题:

Java 有一个 createTempFile() 方法(具有在应用程序终止后删除文件的选项),调用它,然后在 JAR 包中写入要访问此临时文件的文件。然后,您在本地文件系统上拥有 shell 脚本,并且能够执行该脚本。

4

3 回答 3

11

它不是关于 Java,而是关于 shell。据我所知,shell 解释器无法执行驻留在 zip 文件中的脚本。所以你有几个选择:

  1. 从您的 java 程序中将 shell 脚本作为文本文件读取。创建一个临时文件(在临时目录或任何其他相关位置并在那里复制文件的内容。然后用这个临时脚本调用shell解释器。我相信它是最好的方法

  2. 由于 jar 是一个 zip,您可以从 shell 中解压缩它,找到一个脚本并执行 shell。同样,这更尴尬而且相当错误,但从技术上讲它应该可以工作。

  3. 如果您的脚本中没有低级的东西,您可以考虑将逻辑重写为 groovy 脚本(也许您会发现有用的 groosh 项目)。然后从内存而不是文件中调用 groovy 代码。

好吧,我没有想法,如果我是你,我会实施第一个解决方案:) 祝你好运,希望这会有所帮助

于 2012-10-13T17:39:49.330 回答
5

最简单的解决方案是让您的应用程序将 shell 脚本提取到一个临时位置并从那里执行它。

您可以通过调用在 Java 中生成一个临时文件File.createTempFile(String, String)

您可能做的另一件事是将脚本的内容(从您的 jar 加载)作为 shell 的标准输入传递。

为简化起见,如果我有一个a.zip包含脚本的存档a.sh,这将执行它:

$ unzip -p a.zip a.sh | bash

在 Java 程序中,您可以执行以下操作:

import java.io.*;
class Test{

  public static void main(String[]args) throws Exception {

    StringBuilder sb = new StringBuilder();

    // You're not strictly speaking executing a shell script but
    // rather programatically typing commands into the shell
    // sb.append("#!/bin/bash\n");
    sb.append("echo Hello world!\n");
    sb.append("cd /tmp\n");
    sb.append("echo current directory: `pwd`\n");

    // Start the shell
    ProcessBuilder pb = new ProcessBuilder("/bin/bash");
    Process bash = pb.start();

    // Pass commands to the shell
    PrintStream ps = new PrintStream(bash.getOutputStream());
    ps.println(sb);
    ps.close();

    // Get an InputStream for the stdout of the shell
    BufferedReader br = new BufferedReader(
        new InputStreamReader(bash.getInputStream()));

    // Retrieve and print output
    String line;
    while (null != (line = br.readLine())) {
      System.out.println("> "+line);
    }
    br.close();

    // Make sure the shell has terminated, print out exit value
    System.out.println("Exit code: " + bash.waitFor());

  }

}
于 2012-10-13T17:29:32.860 回答
0

我通常只是做...

$ jar xf myapp.jar run.sh && ./run.sh && rm run.sh

...如有必要,您可以从 Runtime.exec() 调用。

于 2012-10-13T18:43:07.687 回答