了解JSch 指令輸出讀取問題
在您提供的使用JSch 的Java 程式碼中,您遇到了程式在嘗試讀取時掛起的情況來自用於捕捉命令輸出的讀取器流。本文解釋了此問題的原因並提供了解決方案。
問題分析
根本問題是由於沒有持續消耗執行命令產生的輸出而產生的。 JSch 指令將輸出累積在內部緩衝區中,如果緩衝區在使用之前已滿,則可能會導致死鎖,命令會掛起等待緩衝區被清除。
解決方案
要解決此問題,您必須連續讀取命令的標準輸出和標準錯誤流,即使在等待命令完成時也是如此。這可確保輸出緩衝區永遠不會填滿並且命令不會掛起。
範例程式碼
以下是包含此解決方案的程式碼的擴充版本:
<code class="java">ChannelExec channel = (ChannelExec)session.openChannel("exec"); channel.setCommand(...); // Set the command to execute ByteArrayOutputStream outputBuffer = new ByteArrayOutputStream(); ByteArrayOutputStream errorBuffer = new ByteArrayOutputStream(); InputStream in = channel.getInputStream(); InputStream err = channel.getExtInputStream(); channel.connect(); byte[] tmp = new byte[1024]; while (true) { while (in.available() > 0) { int i = in.read(tmp, 0, 1024); if (i < 0) break; outputBuffer.write(tmp, 0, i); } while (err.available() > 0) { int i = err.read(tmp, 0, 1024); if (i < 0) break; errorBuffer.write(tmp, 0, i); } if (channel.isClosed()) { if ((in.available() > 0) || (err.available() > 0)) continue; System.out.println("exit-status: " + channel.getExitStatus()); break; } try { Thread.sleep(1000); } catch (Exception ee) { } } System.out.println("output: " + outputBuffer.toString("UTF-8")); System.out.println("error: " + errorBuffer.toString("UTF-8")); channel.disconnect();</code>
在此範例中,while 循環不斷讀取取輸出流(in 和err),同時監視指令的狀態(channel.isClosed())。這可以確保捕獲所有輸出,即使命令產生大量資料也是如此。
以上是為什麼我的 JSch 指令在讀取輸出時掛起?的詳細內容。更多資訊請關注PHP中文網其他相關文章!