Java에서 Linux 쉘 명령을 호출하는 방법


93

리디렉션 (> &) 및 파이프 (|)를 사용하여 Java에서 일부 Linux 명령을 실행하려고합니다. Java는 어떻게 호출 csh하거나 bash명령 할 수 있습니까?

나는 이것을 사용하려고했다 :

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

그러나 리디렉션 또는 파이프와 호환되지 않습니다.


2
cat서로 관련 csh이 없습니다.
Bombe

4
나는 다른 명령에 대한 질문을 이해할 수 있지만 고양이의 경우 : 도대체 왜 파일을 읽지 않습니까?
Atmocreations

8
모든 사람이 처음으로 잘못 이해합니다. Java의 exec ()는 기본 시스템의 쉘을 사용하여 명령을 실행하지 않습니다 (kts가 지적한대로). 리디렉션 및 파이핑은 실제 쉘의 기능이며 Java의 exec ()에서 사용할 수 없습니다.
SteveD

stevendick : 감사합니다. 리디렉션과 배관 때문에 문제가 생겼습니다!
Narek

System.exit (0)은 프로세스가 완료되면 조건부 검사 내부에 있지 않으므로 항상 오류를 출력하지 않고 종료됩니다. 이러한 종류의 실수를 방지하기 위해 중괄호없이 조건문을 작성하지 마십시오.

답변:


97

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. 미안합니다. 나는 여분의 \ " '분명히 그들이 필요하지 않은 (S)를 제거하여 고정 나는 내 시스템에 CSH가없는,하지만 떠들썩한 파티와 함께 작동합니다..
KitsuneYMG

3
다른 사람들이 언급했듯이 이것은 csh 또는 bash가있는 경우 독립적이어야합니다. 그렇지 않습니까?
Narek

@Narek. 그래야하지만 나는 csh가 인수를 처리하는 방법을 모릅니다.
KitsuneYMG

1
감사합니다. 이것은 효과가 있었다. 사실 저는 "csh"대신 "sh"를 사용했습니다.
Farshid

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. * 이후에도; 제대로 가져 오기 위의 예제를 실행할 수 없습니다.
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 개의 백 슬래시가 필요한지 말해 줄 수 있다면 뭔가를 배울 수 있습니다. 내가 계산하는 것보다 한 단계 더 많은 탈출이 일어나지 않습니다.

편집 : Linux에서이 동일한 코드를 시도한 결과 테스트 명령에 백 슬래시가 절반으로 필요하다는 것이 밝혀졌습니다! (즉, 예상되는 숫자는 2와 4입니다.) 이제 더 이상 이상하지 않고 이식성 문제입니다.


답변의 일부가 아닌 새로운 StackOverflow 질문으로 질문해야합니다.
vog

OSX / Oracle java8에서 2 개 및 4 개와 함께 작동합니다. (당신의 성격을 언급하지 않음) 원래 자바 환경에 문제가있는 것 같다
TheMadsen
당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.