轉自:調用Process.waitfor致使的進程掛起 html
最近遇到pipe_wait問題,父進程調用子進程時,子進程阻塞,cat /proc/$child/wchan輸出pipe_wait,進程阻塞在pipe_wait不執行,轉載文章對此問題分析很透徹。java
若是要在Java中調用shell腳本時,可使用Runtime.exec或ProcessBuilder.start。它們都會返回一個Process對象,經過這個Process能夠對獲取腳本執行的輸出,而後在Java中進行相應處理。例如,下面的代碼:node
一般,安全編碼規範中都會指出:使用Process.waitfor的時候,可能致使進程阻塞,甚至死鎖。 那麼這句應該怎麼理解呢?用個實際的例子說明下。linux
使用Java代碼調用shell腳本,執行後會發現Java進程和Shell進程都會掛起,沒法結束。shell
Java代碼 processtest.javac#
被調用的Shell腳本doecho.sh安全
基於上述分析,只要主進程在waitfor以前,能不斷處理緩衝區中的數據就能夠。由於,咱們能夠再waitfor以前,單獨啓兩個額外的線程,分別用於處理InputStream和ErrorStream就能夠。實例代碼以下:bash
By default, the created subprocess does not have its own terminal or console. All its standard I/O (i.e. stdin, stdout, stderr) operations will be redirected to the parent process, where they can be accessed via the streams obtained using the methods getOutputStream(), getInputStream(), and getErrorStream(). The parent process uses these streams to feed input to and get output from the subprocess. Because some native platforms only provide limited buffer size for standard input and output streams, failure to promptly write the input stream or read the output stream of the subprocess may cause the subprocess to block, or even deadlock.electron
從JDK的說明中能夠看出兩點:async
要回答上面的問題能夠從系統的層面嘗試分析。
首先經過ps命令能夠看到,在Linux上多出了兩個進程:一個Java進程、一個shell進程,且shell是java的子進程。
而後,能夠看到shell進程的狀態顯示爲pipe_w。我剛開始覺得pipe_w表示pipe_write。進一步查看/proc/pid/wchan 發現pipe_w其實表示爲pipe_wait。一般/proc/pid/wchan表示一個內存地址或進程正在執行的方法名稱。所以,這彷佛代表該進程 在操做pipe時發生了等待,從而被掛起。咱們知道pipe是IPC的一種,一般用於父子進程之間通訊。這樣咱們能夠猜想:多是父子進程之間經過 pipe通訊的時候出現了阻塞。
另外,觀察父子進程的fd信息,即/proc/pid/fd。能夠看到子進程的0/1/2(即:stdin/stdout/stderr)分別被重定向到了三個pipe文件;父親進程中對應的也有對着三個pipe文件的引用。
綜上所述,這個過程應該是這樣的:子進程不斷向pipe中寫數據,而父進程一直不讀取pipe中的數據,致使pipe被塞滿,子進程沒法繼續寫入,因此出現pipe_wait的狀態。那麼pipe到底有多大呢?
由於我已經在doecho.sh的腳步中記錄了打印了字符數,查看count.log就能夠知道子進程最終發送了多少數據。在子進程掛起 了,count.log的數據一致保持在6543不變。故,當前子進程向pipe中寫入6543*10=65430bytes時,出現進程掛起。 65536-65430=106byte即距離64K差了106bytes。
換另外的測試方式,每次寫入1k,記錄總共能夠寫入多少。進程代碼如test_pipe_size.sh所示。測試結果爲64K。兩次結果相差了106byte,那個這個pipe到底多大?
最直接的方式就是看源碼。Pipe的實現代碼主要在linux/fs/pipe.c中,咱們主要看pipe_wait方法。
Java 中的進程與線程
https://www.ibm.com/developerworks/cn/java/j-lo-processthread/
When Runtime.exec() won't
http://www.javaworld.com/article/2071275/core-java/when-runtime-exec---won-t.html?page=3
Linux進程間通訊之管道(pipe)、命名管道(FIFO)與信號(Signal)
http://www.cnblogs.com/biyeymyhjob/archive/2012/11/03/2751593.html
buffering in standard streams
http://www.pixelbeat.org/programming/stdio_buffering/
Todd.log - a place to keep my thoughts onprogramming
http://www.cnblogs.com/weidagang2046/p/io-redirection.html
linux cross reference
http://lxr.free-electrons.com/source/fs/pipe.c#L103
How big is the pipe buffer
http://unix.stackexchange.com/questions/11946/how-big-is-the-pipe-buffer