如何从Java调用Linux Shell命令


93

我正在尝试使用重定向(>&)和管道(|)从Java执行一些Linux命令。Java如何调用cshbash命令?

我试图用这个:

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

但这与重定向或管道不兼容。


2
cat而且csh彼此之间没有任何关系
孟买

4
我可以理解其他命令的问题,但对于猫来说:为什么你不读文件呢?
Atmocreations 2009年

8
每个人第一次都会遇到这种错误-Java的exec()并不使用底层系统的Shell执行命令(如kts所指出的)。重定向和管道传递是真正的shell的功能,Java的exec()无法使用它们。
SteveD

stevendick:非常感谢,由于重定向和管道传输,我遇到了问题!
Narek

System.exit(0)不在条件检查中,如果进程已完成,则它将始终退出而不会输出错误。切勿在没有大括号的情况下写条件语句,以免发生此类错误。

Answers:


96

exec在您的shell中不执行命令

尝试

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 2009年

3
正如其他人提到的,如果您拥有csh或bash,这应该是独立的,不是吗?
Narek

@Narek。应该,但我不知道;不知道csh如何处理参数。
KitsuneYMG

1
谢谢。这工作了。实际上,我使用“ sh”而不是“ csh”。
Farshid 2012年

5
警告:此解决方案很可能会遇到其挂起的典型问题,因为您没有阅读其输出和错误流。例如:stackoverflow.com/questions/8595748/java-runtime-exec
Evgeni Sergeev

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。*之后;已正确导入,我无法获得上面的示例来运行。
史蒂夫·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);

    }
}

(测试示例是 命令,命令按时间顺序递归列出目录及其子目录中的所有文件。)

顺便说一句,如果有人可以告诉我为什么我在那里需要四个和八个反斜杠,而不是两个和四个,我可以学到一些东西。比我所计数的还要多的一种无法逃避的事情正在发生。

编辑:刚刚在Linux上尝试过相同的代码,结果发现我在test命令中需要一半的反斜杠!(即:预期的2和4。)现在,它不再只是怪异的,而是可移植性问题。


您应该将问题作为新的StackOverflow问题提出,而不应作为回答的一部分。
vog

在OSX / Oracle java8上可以使用两个和四个。您的原始Java环境似乎有问题(您没有提及其本质)
TheMadsen
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.