วิธีเรียกใช้คำสั่ง Linux shell จาก Java


93

ฉันพยายามเรียกใช้คำสั่ง Linux บางคำสั่งจาก Java โดยใช้การเปลี่ยนเส้นทาง (> &) และไปป์ (|) Java เรียกใช้cshหรือbashคำสั่งได้อย่างไร?

ฉันพยายามใช้สิ่งนี้:

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

แต่ไม่สามารถใช้งานได้กับการเปลี่ยนเส้นทางหรือไปป์


2
catและcshไม่มีส่วนเกี่ยวข้องใด ๆ
Bombe

4
ฉันเข้าใจคำถามสำหรับคำสั่งอื่น ๆ แต่สำหรับ cat: ทำไมคุณไม่อ่านในไฟล์?
Atmocreations

8
ทุกคนทำผิดในครั้งแรก - exec () ของ Java ไม่ใช้เชลล์ของระบบพื้นฐานเพื่อดำเนินการคำสั่ง (ตามที่ kts ชี้ให้เห็น) การเปลี่ยนเส้นทางและการวางท่อเป็นคุณสมบัติของเชลล์จริงและไม่สามารถใช้งานได้จาก exec () ของ Java
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"});

@ นเรก. ขอโทษด้วยกับเรื่องนั้น. ฉันแก้ไขโดยการลบ "พิเศษ" เห็นได้ชัดว่าพวกเขาไม่จำเป็นฉันไม่มี csh ในระบบของฉัน แต่ใช้งานได้กับ bash
KitsuneYMG

3
อย่างที่คนอื่นพูดถึงสิ่งนี้ควรเป็นอิสระโดยที่คุณมี csh หรือ bash ไม่ใช่เหรอ
Narek

@ นเรก. ควร แต่ฉันไม่รู้ว่า csh จัดการกับข้อโต้แย้งอย่างไร
KitsuneYMG

1
ขอบคุณ. สิ่งนี้ได้ผล อันที่จริงฉันใช้ "sh" แทน "csh"
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 ฉันได้อัปเดตโค้ดตัวอย่างด้านบนเพื่อลบคำสั่งการบันทึกและตัวแปรที่ประกาศนอกโค้ดที่วางไว้และตอนนี้ควรคอมไพล์และรัน อย่าลังเลที่จะลองและแจ้งให้เราทราบว่ามันทำงานอย่างไร
ทิม

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 และปรากฎว่าฉันต้องการแบ็กสแลชครึ่งหนึ่งในคำสั่งทดสอบ! (นั่นคือจำนวนที่คาดไว้คือสองและสี่) ตอนนี้ไม่ใช่แค่เรื่องแปลกอีกต่อไป แต่เป็นปัญหาในการพกพา


คุณควรถามคำถามของคุณเป็นคำถาม StackOverflow ใหม่ไม่ใช่เป็นส่วนหนึ่งของคำตอบของคุณ
vog

ทำงานร่วมกับสองและสี่บน OSX / Oracle java8 ดูเหมือนว่าจะมีปัญหากับสภาพแวดล้อม java ดั้งเดิมของคุณ (ซึ่งคุณไม่ได้กล่าวถึงลักษณะของ)
TheMadsen
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.