JavaからLinuxシェルコマンドを呼び出す方法


93

リダイレクト(>&)とパイプ(|)を使用してJavaからいくつかのLinuxコマンドを実行しようとしています。Javaはどのように起動cshまたはbashコマンドできますか?

私はこれを使ってみました:

Process p = Runtime.getRuntime().exec("shell command");

ただし、リダイレクトやパイプとは互換性がありません。


2
catcshお互いに何の関係もありません。
ボンベ

4
私は他のコマンドの質問を理解できますが、猫の場合:なぜ地獄にファイルを読み取らないのですか?
Atmocreations 2009

8
誰もが最初にこれを間違えます-Javaのexec()は、基になるシステムのシェルを使用してコマンドを実行しません(ktsが指摘しているように)。リダイレクトとパイピングは実際のシェルの機能であり、Javaのexec()からは利用できません。
SteveD

stevendick:ありがとうございました。リダイレクトとパイプのために問題が発生しました!
Narek、

System.exit(0)は、プロセスが完了した場合の条件チェックの内部にないため、エラーを出力せずに常に終了します。この種の間違いを正確に回避するために、中括弧なしで条件文を記述しないでください。

回答:


96

execはシェルでコマンドを実行しません

試す

Process p = Runtime.getRuntime().exec(new String[]{"csh","-c","cat /home/narek/pk.txt"});

代わりに。

編集::私のシステムにはcshがないため、代わりにbashを使用しました。以下は私のために働きました

Process p = Runtime.getRuntime().exec(new String[]{"bash","-c","ls /home/XXX"});

@Narek。申し訳ありません。私は、余分な\」さんどうやら、彼らが必要とされていないを削除することによってそれを修正し、私は私のシステムでcshのを持っていないが、それはbashので動作します。。
KitsuneYMG

3
他の人がcshまたはbashを持っているかどうかに関係なくこれは独立しているはずだと述べたように、そうではありませんか?
Narek、

@Narek。cshが引数をどのように処理するかはわかりませんが、わかりません。
KitsuneYMG 2009

1
ありがとうございました。これはうまくいきました。実際には「csh」の代わりに「sh」を使用しました。
Farshid

5
警告:このソリューションは、出力とエラーストリームを読み取っていないため、ハングするという一般的な問題に遭遇する可能性が非常に高くなります。例:stackoverflow.com/questions/8595748/java-runtime-exec
Evgeni Sergeev 2014

32

ProcessBuilderを使用して、スペースではなくコマンドと引数を区切ります。これは、使用するシェルに関係なく機能するはずです。

import java.io.BufferedReader;
import java.io.File;
import java.io.IOException;
import java.io.InputStreamReader;
import java.util.ArrayList;
import java.util.List;

public class Test {

    public static void main(final String[] args) throws IOException, InterruptedException {
        //Build command 
        List<String> commands = new ArrayList<String>();
        commands.add("/bin/cat");
        //Add arguments
        commands.add("/home/narek/pk.txt");
        System.out.println(commands);

        //Run macro on target
        ProcessBuilder pb = new ProcessBuilder(commands);
        pb.directory(new File("/home/narek"));
        pb.redirectErrorStream(true);
        Process process = pb.start();

        //Read output
        StringBuilder out = new StringBuilder();
        BufferedReader br = new BufferedReader(new InputStreamReader(process.getInputStream()));
        String line = null, previous = null;
        while ((line = br.readLine()) != null)
            if (!line.equals(previous)) {
                previous = line;
                out.append(line).append('\n');
                System.out.println(line);
            }

        //Check result
        if (process.waitFor() == 0) {
            System.out.println("Success!");
            System.exit(0);
        }

        //Abnormal termination: Log command parameters and output and throw ExecutionException
        System.err.println(commands);
        System.err.println(out.toString());
        System.exit(1);
    }
}

java.util。*;の後でも 上記の例を実行できません。
Steve K

@Stephan上記のサンプルコードを更新して、貼り付けられたコードの外部で宣言されたロギングステートメントと変数を削除しました。これで、コンパイルと実行ができるはずです。試してみて、どのように機能するか教えてください。
Tim

15

自己完結型メソッドを作成するための@Timの例に基づいて構築します。

import java.io.BufferedReader;
import java.io.File;
import java.io.InputStreamReader;
import java.util.ArrayList;

public class Shell {

    /** Returns null if it failed for some reason.
     */
    public static ArrayList<String> command(final String cmdline,
    final String directory) {
        try {
            Process process = 
                new ProcessBuilder(new String[] {"bash", "-c", cmdline})
                    .redirectErrorStream(true)
                    .directory(new File(directory))
                    .start();

            ArrayList<String> output = new ArrayList<String>();
            BufferedReader br = new BufferedReader(
                new InputStreamReader(process.getInputStream()));
            String line = null;
            while ( (line = br.readLine()) != null )
                output.add(line);

            //There should really be a timeout here.
            if (0 != process.waitFor())
                return null;

            return output;

        } catch (Exception e) {
            //Warning: doing this is no good in high quality applications.
            //Instead, present appropriate error messages to the user.
            //But it's perfectly fine for prototyping.

            return null;
        }
    }

    public static void main(String[] args) {
        test("which bash");

        test("find . -type f -printf '%T@\\\\t%p\\\\n' "
            + "| sort -n | cut -f 2- | "
            + "sed -e 's/ /\\\\\\\\ /g' | xargs ls -halt");

    }

    static void test(String cmdline) {
        ArrayList<String> output = command(cmdline, ".");
        if (null == output)
            System.out.println("\n\n\t\tCOMMAND FAILED: " + cmdline);
        else
            for (String line : output)
                System.out.println(line);

    }
}

(テスト例は、ディレクトリとそのサブディレクトリ内のすべてのファイルを時系列で再帰的に一覧表示するコマンドです。)

ちなみに、なぜ2つや4つではなく4つや8つのバックスラッシュが必要なのかを誰かが教えてくれれば、何かを学ぶことができます。私が数えているよりも、エスケープ解除の発生のレベルが1つあります。

編集:Linuxで同じコードを試したところ、テストコマンドに必要なバックスラッシュの数が半分になることがわかりました。(つまり、予想される数は2と4です。)変なだけではなく、移植性の問題です。


質問は、回答の一部としてではなく、新しいStackOverflowの質問として行う必要があります。
2016

OSX / Oracle java8で2および4で動作します。元のJava環境に問題があるようです(その性質については触れていません)
TheMadsen
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.