Java程序執(zhí)行Cmd指令所遇問題記錄及解決方案
這篇是有關(guān)在編寫Java程序執(zhí)行Cmd指令時(shí)所遇到的問題記錄,其中有一些是個(gè)人的理解,如有問題望不吝賜教,感謝❤
Windows 命令提示符(cmd.exe)是 Windows NT 下的一個(gè)用于運(yùn)行 Windows 控制面板程序或某些 DOS 程序的shell程序
1.執(zhí)行Cmd命令的兩種方式
(1)RunTime.getRunTime().exec(多種重載方式) - 會(huì)得到一個(gè)Process對(duì)象通過其start()方法開啟一個(gè)新進(jìn)程以執(zhí)行輸入的指令。

這種方法就不多說了,最后這種形式還是用到第二種方式的方法(Java Api文檔中也推薦使用第二種方式去創(chuàng)建一個(gè)Process對(duì)象):

* @see ProcessBuilder
* @since 1.3
*/
public Process exec(String[] cmdarray, String[] envp, File dir)
throws IOException {
return new ProcessBuilder(cmdarray)
.environment(envp)
.directory(dir)
.start();
}
(2).new ProcessBuilder().command(指令)
2.獲取執(zhí)行指令后的輸出:

在這里就遇到點(diǎn)問題,
上面兩種方式執(zhí)行Windows自帶的命令都沒有什么問題(像Ping、Ipconfig)。但是當(dāng)執(zhí)行像“Java -version”這樣的外部命令,其輸出通過getInputStream()方法是拿不到的。
后來是通過參考網(wǎng)上資料,采用將子進(jìn)程的輸出重定向到文件中,再從文件中讀取內(nèi)容的方法:
// 外部程序的輸出放到了錯(cuò)誤信息輸出流中,不將錯(cuò)誤信息流輸出到文件話,輸出信息就看不到了😂
pb.redirectErrorStream(true);
// 把執(zhí)行結(jié)果輸出
pb.redirectOutput(file);
//等待語句執(zhí)行完成,否則可能會(huì)讀不到結(jié)果。
pb.start().waitFor();
InputStream in = new FileInputStream(file);
br = new BufferedReader(new InputStreamReader(in,charsetName));
String line = null;
while ((line = br.readLine()) != null) {
outPutResult.append(line).append("\n");
}
br.close();
br = null;
// 刪除臨時(shí)文件
file.delete();
最新解決方法:剛寫完這篇博客,就在想Java開發(fā)文檔中這句“否則,如果使用ProcessBuilder.redirectErrorStream重定向子進(jìn)程的標(biāo)準(zhǔn)錯(cuò)誤,則此方法返回的輸入流將接收合并的標(biāo)準(zhǔn)輸出和子進(jìn)程的標(biāo)準(zhǔn)錯(cuò)誤。”(下面圖片)怎么就沒用呢,結(jié)果回頭一看,文檔是Java 8的,我跑的程序用的是Java 7的,把自己整笑了🙃,還在這一通瞎操作。
而至于為什么要將子進(jìn)程標(biāo)準(zhǔn)輸出和子進(jìn)程的標(biāo)準(zhǔn)錯(cuò)誤輸出合并,可以看下小弟下面的拙見。
對(duì)于非Windows自帶命令,可以這樣寫(不再需要借助文件):
public static StringBuilder runOutCmdTest(String command) {
BufferedReader br = null;
StringBuilder outPutResult = new StringBuilder();
try{
ProcessBuilder pb = new ProcessBuilder().command("cmd.exe", "/c", command);
// 外部程序的輸出放到了錯(cuò)誤信息輸出流中
pb.redirectErrorStream(true);
// 等待語句執(zhí)行完成,否則可能會(huì)讀不到結(jié)果。
Process process = pb.start();
process.waitFor();
InputStream inputStream = process.getInputStream();
br = new BufferedReader(new InputStreamReader(inputStream, "GBK"));
String line;
while ((line = br.readLine()) != null) {
outPutResult.append(line).append("\n");
}
br.close();
br = null;
} catch (Exception e) {
e.printStackTrace();
}
return outPutResult;
}
3.關(guān)于getInputStream ()無法得到子進(jìn)程輸出的原因

此方法獲取的流是子進(jìn)程正常輸出流不包括異常錯(cuò)誤信息流,Process對(duì)象將異常信息放在了ErrorStream中。這里可以試一下,會(huì)發(fā)現(xiàn)執(zhí)行“Java -version”控制臺(tái)輸出的是紅字,也就是異常信息。

emmm至于為什么非Windows自帶命令的正常輸出會(huì)被視作異常信息,不太清楚,下次再looklook源碼。
而按上面圖片的最后一句,
4.一個(gè)調(diào)用指令例子
private static final String TEMP_FILE_PATH = "D:\\temp.txt";
/**
* 運(yùn)行外部程序命令 無參數(shù)時(shí)調(diào)用
* @param command 輸入命令
* @return 輸出內(nèi)容
*/
public static StringBuilder runOutCmd(String command) {
// 默認(rèn)字符解析GBK
return runOutCmd(command, null,"GBK");
}
/**
* 運(yùn)行外部程序命令 帶參數(shù)
* @param command 輸入命令
* @param args 輸入?yún)?shù)
* @return
*/
public static StringBuilder runOutCmd(String command, List<String> args) {
// 默認(rèn)字符解析GBK
return runOutCmd(command, args,"GBK");
}
/**
* 運(yùn)行外部程序命令 - 帶參數(shù)并規(guī)定字符解析格式
* @param args 輸入?yún)?shù)
* @param command 輸入命令
* @param charsetName 輸出字符解析格式
* @return
*/
public static StringBuilder runOutCmd(String command, List<String> args, String charsetName) {
BufferedReader br = null;
StringBuilder outPutResult = new StringBuilder();
try {
// 新建一個(gè)用來存儲(chǔ)子進(jìn)程輸出結(jié)果結(jié)果的緩存文件
File file = new File(TEMP_FILE_PATH);
if (!file.getParentFile().exists()) {
file.getParentFile().mkdirs();
}
if (!file.exists()) {
file.createNewFile();
}
List<String> execCommand = new LinkedList<>();
if (args != null) {
execCommand.addAll(args);
}
execCommand.add(0,command);
execCommand.add(0,"/c");
execCommand.add(0,"cmd.exe");
ProcessBuilder pb = new ProcessBuilder().command(execCommand).inheritIO();
// 外部程序的輸出放到了錯(cuò)誤信息輸出流中
pb.redirectErrorStream(true);
// 把執(zhí)行結(jié)果輸出
pb.redirectOutput(file);
//等待語句執(zhí)行完成,否則可能會(huì)讀不到結(jié)果。
pb.start().waitFor();
InputStream in = new FileInputStream(file);
br = new BufferedReader(new InputStreamReader(in,charsetName));
String line = null;
while ((line = br.readLine()) != null) {
outPutResult.append(line).append("\n");
}
br.close();
br = null;
// 刪除臨時(shí)文件
file.delete();
} catch (Exception e) {
e.printStackTrace();
} finally {
if (br != null) {
try {
br.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
return outPutResult;
}
最新例子(不用借助文件):
/**
* 運(yùn)行外部程序命令 - 帶參數(shù)并規(guī)定字符解析格式
*
* @param args 輸入?yún)?shù)
* @param command 輸入命令
* @param charsetName 輸出字符解析格式
* @return
*/
public static StringBuilder runOutCmd(String command, List<String> args, String charsetName) {
BufferedReader br = null;
StringBuilder outPutResult = new StringBuilder();
try {
List<String> execCommand = new LinkedList<>();
if (args != null) {
execCommand.addAll(args);
}
execCommand.add(0, command);
execCommand.add(0, "/c");
execCommand.add(0, "cmd.exe");
ProcessBuilder pb = new ProcessBuilder().command(execCommand).inheritIO();
// 外部程序的輸出放到了錯(cuò)誤信息輸出流中
pb.redirectErrorStream(true);
//等待語句執(zhí)行完成,否則可能會(huì)讀不到結(jié)果。
Process process = pb.start();
process.waitFor();
InputStream inputStream = process.getInputStream();
br = new BufferedReader(new InputStreamReader(inputStream, charsetName));
String line;
while ((line = br.readLine()) != null) {
outPutResult.append(line).append("\n");
}
br.close();
br = null;
} catch (Exception e) {
e.printStackTrace();
} finally {
if (br != null) {
try {
br.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
return outPutResult;
}
ps:
1. inheritIO()作用:

意味著使用此方法,子進(jìn)程的報(bào)錯(cuò)的異常信息也會(huì)在當(dāng)前Java進(jìn)程的控制臺(tái)輸出,而Process對(duì)象將非Windows命令的輸出視為異常信息,那么非Windows命令的輸出當(dāng)使用了此方法的時(shí)候會(huì)在控制臺(tái)輸出。
2.字符解析格式問題:
輸出出現(xiàn)亂碼,與Cmd程序字符格式默認(rèn)為GBK有關(guān)。
按實(shí)際情況修改流的解析格式就可以了:
br = new BufferedReader(new InputStreamReader(in,charsetName));

到此這篇關(guān)于Java程序執(zhí)行Cmd指令所遇問題記錄及解決方案的文章就介紹到這了,更多相關(guān)Java程序執(zhí)行Cmd指令內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
SpringCloud解決feign調(diào)用token丟失問題解決辦法
在feign調(diào)用中可能會(huì)遇到如下問題:同步調(diào)用中,token丟失,這種可以通過創(chuàng)建一個(gè)攔截器,將token做透傳來解決,異步調(diào)用中,token丟失,這種就無法直接透傳了,因?yàn)樽泳€程并沒有token,這種需要先將token從父線程傳遞到子線程,再進(jìn)行透傳2024-05-05
SpringSecurity動(dòng)態(tài)加載用戶角色權(quán)限實(shí)現(xiàn)登錄及鑒權(quán)功能
這篇文章主要介紹了SpringSecurity動(dòng)態(tài)加載用戶角色權(quán)限實(shí)現(xiàn)登錄及鑒權(quán)功能,很多朋友感覺這個(gè)功能很難,今天小編通過實(shí)例代碼給大家講解,需要的朋友可以參考下2019-11-11
SpringBoot集成RabbitMQ的方法(死信隊(duì)列)
這篇文章主要介紹了SpringBoot集成RabbitMQ的方法(死信隊(duì)列),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-05-05
springboot項(xiàng)目啟動(dòng)類錯(cuò)誤(找不到或無法加載主類 com.**Application)
本文主要介紹了spring-boot項(xiàng)目啟動(dòng)類錯(cuò)誤(找不到或無法加載主類 com.**Application),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2024-05-05
SpringBoot多模塊搭建的實(shí)現(xiàn)示例
多模塊開發(fā)是指將一個(gè)大型應(yīng)用程序拆分為多個(gè)模塊,本文主要介紹了SpringBoot多模塊搭建的實(shí)現(xiàn)示例,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2024-08-08
public?static?void?main(String[]?args)使用解讀
這篇文章主要介紹了public?static?void?main(String[]?args)的使用,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-01-01
Java實(shí)現(xiàn)文件名倒序排序的技術(shù)指南
在實(shí)際開發(fā)過程中,我們經(jīng)常需要對(duì)文件進(jìn)行操作和處理,一個(gè)常見的需求是按文件名倒序排列文件列表,以便于文件的管理和查找,本文將介紹如何在Java中實(shí)現(xiàn)文件名倒序排序,并提供詳細(xì)的代碼案例,需要的朋友可以參考下2024-08-08

