Informationen zu Leseproblemen bei JSch-Befehlsausgaben
In Ihrem bereitgestellten Java-Code mit JSch kommt es zu einer Situation, in der das Programm beim Leseversuch hängen bleibt aus einem Reader-Stream, der zum Erfassen der Befehlsausgabe verwendet wird. In diesem Artikel wird die Ursache dieses Problems erläutert und eine Lösung bereitgestellt.
Problemanalyse
Das grundlegende Problem entsteht dadurch, dass die durch den ausgeführten Befehl erzeugte Ausgabe nicht kontinuierlich verbraucht wird. JSch-Befehle sammeln die Ausgabe in einem internen Puffer, und wenn der Puffer voll ist, bevor er verbraucht wird, kann es zu einem Deadlock kommen, bei dem der Befehl hängen bleibt und darauf wartet, dass der Puffer gelöscht wird.
Lösung
Um dieses Problem zu beheben, müssen Sie kontinuierlich sowohl die Standardausgabe als auch die Standardfehlerströme des Befehls lesen, auch während Sie auf den Abschluss des Befehls warten. Dadurch wird sichergestellt, dass der Ausgabepuffer nie voll wird und der Befehl nicht hängen bleibt.
Beispielcode
Hier ist eine erweiterte Version Ihres Codes, die diese Lösung enthält:
<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>
In diesem Beispiel lesen die While-Schleifen kontinuierlich die Ausgabeströme (in und err) und überwachen gleichzeitig den Status des Befehls (channel.isClosed()). Dadurch wird sichergestellt, dass die gesamte Ausgabe erfasst wird, auch wenn der Befehl eine große Datenmenge erzeugt.
Das obige ist der detaillierte Inhalt vonWarum hängt mein JSch-Befehl beim Lesen der Ausgabe?. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!