基于Process#waitFor()阻塞問題的解決
Process#waitFor()阻塞問題
有時(shí)需要在程序中調(diào)用可執(zhí)行程序或腳本命令:
Process process = Runtime.getRuntime().exec(shPath); int exitCode = process .waitFor();
Runtime.getRuntime()返回當(dāng)前應(yīng)用程序的Runtime對象,該對象的exec()方法指示Java虛擬機(jī)創(chuàng)建一個(gè)子進(jìn)程執(zhí)行指定的可執(zhí)行程序,并返回與該子進(jìn)程對應(yīng)的Process對象實(shí)例。通過Process可以控制該子進(jìn)程的執(zhí)行或獲取該子進(jìn)程的信息。
它的所有標(biāo)準(zhǔn)io(即stdin,stdout,stderr)操作都將通過三個(gè)流(getOutputStream(),getInputStream(),getErrorStream())重定向到父進(jìn)程。父進(jìn)程使用這些流來提供到子進(jìn)程的輸入和獲得從子進(jìn)程的輸出。因?yàn)檩斎牒洼敵隽魈峁┯邢薜木彌_區(qū)大小,如果讀寫子進(jìn)程的輸出流或輸入流出現(xiàn)失敗,當(dāng)緩沖區(qū)滿之后將無法繼續(xù)寫入數(shù)據(jù),則可能導(dǎo)致子進(jìn)程阻塞,最終造成阻塞在waifor()這里。
針對這種情況,解法是要清空getInputStream()和getErrorStream()這兩個(gè)流。而且兩個(gè)流的清空一定是異步的。
static void drainInBackground(final InputStream is) { new Thread(new Runnable(){ public void run(){ try{ while( is.read() >= 0 ); } catch(IOException e){ // return on IOException } } }).start(); }
還有一種解法是用ProcessBuilder來創(chuàng)建Process對象,必須要使用ProcessBuilder的redirectErrorStream方法。redirectErrorStream方法設(shè)置為ture的時(shí)候,會(huì)將getInputStream(),getErrorStream()兩個(gè)流合并,自動(dòng)會(huì)清空流,無需自己處理。如果是false,getInputStream(),getErrorStream()兩個(gè)流分開,就必須自己處理,程序如下:
try { ProcessBuilder pbuilder=new ProcessBuilder("ping","192.168.0.125"); pbuilder.redirectErrorStream(true); process=pbuilder.start(); reader=new BufferedReader(new InputStreamReader(process.getInputStream())); String line=null; while((line=reader.readLine())!=null){ System.out.println(line); } int result=process.waitFor(); System.out.println(result); } catch (IOException e) { // TODO Auto-generated catch block e.printStackTrace(); } catch (InterruptedException e) { // TODO Auto-generated catch block e.printStackTrace(); }
Process process = null; try { process = launchProcess(cmdlist, environment); StringBuilder sb = new StringBuilder(); String output = getOutput(process.getInputStream()); String errorOutput = getOutput(process.getErrorStream()); sb.append(output); sb.append("\n"); sb.append(errorOutput); boolean ret = process.waitFor(1L, TimeUnit.SECONDS); if (!ret) { System.out.println(command + " is terminated abnormally. ret={}, str={}" + ret + " " + sb.toString()); } return sb.toString(); } catch (Throwable e) { System.out.println("Failed to run " + command + ", "); e.printStackTrace(); } finally { if (null != process) { process.destroy(); } } return "";
注意process一定要釋放
Process.waitFor()導(dǎo)致主線程堵塞
今日開發(fā)的時(shí)候使用jdk自帶的運(yùn)行時(shí)變量 RunTime.getRunTime() 去執(zhí)行bash命令。
因?yàn)樵揵ash操作耗時(shí)比較長,所以使用了Process.waitFor()去等待子線程運(yùn)行結(jié)束。
這個(gè)時(shí)候發(fā)現(xiàn)程序卡在waitFor()沒有繼續(xù)往下執(zhí)行。
看了官方解釋:
waitFor
:等待子進(jìn)程執(zhí)行結(jié)束,或者已終止子進(jìn)程,此方法立即返回。
當(dāng)RunTime對象調(diào)用exec方法后,jvm會(huì)創(chuàng)建一個(gè)子進(jìn)程,該子進(jìn)程與jvm建立三個(gè)管道連接:標(biāo)準(zhǔn)輸入流、標(biāo)準(zhǔn)輸出流、標(biāo)準(zhǔn)錯(cuò)誤流。假設(shè)該子進(jìn)程不斷向標(biāo)準(zhǔn)輸入流、標(biāo)準(zhǔn)輸出流寫數(shù)據(jù),而jvm不讀取的話,會(huì)導(dǎo)致緩沖區(qū)塞滿而無法繼續(xù)寫數(shù)據(jù),最終堵塞在waitFor這里。
知道了問題所在就好處理了, 我們只需要將子進(jìn)程返回的信息從緩沖區(qū)讀取出來,便可以避免主線程堵塞問題。
public static void main(String[] args){ Process proc = RunTime.getRunTime().exec("sh /home/winnie/dataExtract.sh") // 標(biāo)準(zhǔn)輸入流(必須寫在 waitFor 之前) String inStr = consumeInputStream(proc.getInputStream()); // 標(biāo)準(zhǔn)錯(cuò)誤流(必須寫在 waitFor 之前) String errStr = consumeInputStream(proc.getErrorStream()); int retCode = proc.waitFor(); if(retCode == 0){ System.out.println("程序正常執(zhí)行結(jié)束"); } } /** * 消費(fèi)inputstream,并返回 */ public static String consumeInputStream(InputStream is){ BufferedReader br = new BufferedReader(new InputStreamReader(is)); String s ; StringBuilder sb = new StringBuilder(); while((s=br.readLine())!=null){ System.out.println(s); sb.append(s); } return sb.toString(); }
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
關(guān)于SpringBoot微服務(wù)發(fā)布與部署的三種方式
SpringBoot 框架只提供了一套基于可執(zhí)行 jar 包(executable jar)格式的標(biāo)準(zhǔn)發(fā)布形式,但并沒有對部署做過多的界定,而且為了簡化可執(zhí)行 jar 包的生成,SpringBoot 提供了相應(yīng)的 Maven 項(xiàng)目插件,需要的朋友可以參考下2023-05-05Java日常練習(xí)題,每天進(jìn)步一點(diǎn)點(diǎn)(46)
下面小編就為大家?guī)硪黄狫ava基礎(chǔ)的幾道練習(xí)題(分享)。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧,希望可以幫到你2021-08-08DUBBO 日志過濾器,輸出dubbo 接口調(diào)用入?yún)?、出參等信?最新推薦)
這篇文章主要介紹了DUBBO 日志過濾器,輸出dubbo 接口調(diào)用入?yún)?、出參等信?首先自定義一個(gè)過濾器?DubboLoggerFilter.java,本文結(jié)合示例代碼給大家講解的非常詳細(xì),需要的朋友可以參考下2022-12-12IDEA中Java出現(xiàn)無效的源發(fā)行版錯(cuò)誤的解決辦法
這篇文章主要給大家介紹了關(guān)于IDEA中Java出現(xiàn)無效的源發(fā)行版錯(cuò)誤的解決辦法,IDEA中Java出現(xiàn)?效的源發(fā)?版解決辦法出現(xiàn)該問題的原因是項(xiàng)?Project當(dāng)中的jdk與電腦當(dāng)中的jdk版本不?致造成的,需要的朋友可以參考下2023-10-10java String 轉(zhuǎn)成Double二維數(shù)組的方法
下面小編就為大家?guī)硪黄猨ava String 轉(zhuǎn)成Double二維數(shù)組的方法。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2016-10-10Java調(diào)用ChatGPT(基于SpringBoot和Vue)實(shí)現(xiàn)可連續(xù)對話和流式輸出的ChatGPT API
這篇文章主要介紹了Java調(diào)用ChatGPT(基于SpringBoot和Vue),實(shí)現(xiàn)可連續(xù)對話和流式輸出的ChatGPT API(可自定義實(shí)現(xiàn)AI助手),文中代碼示例介紹的非常詳細(xì),感興趣的朋友可以參考下2023-04-04