PrintWriter.print 字符串太长时阻塞
PrintWriter.print blocks when the string is too long
我正在 java 中编写一个包装程序,它只是应该通过写入其他进程的标准 in 流来将参数传递给其他进程,并从他们的标准输出流读取响应。但是,当我尝试传入的 String
太大时,PrintWriter.print
只会阻塞。没有错误,只是冻结。对此有好的解决方法吗?
相关代码
public class Wrapper {
PrintWriter writer;
public Wrapper(String command){
start(command);
}
public void call(String args){
writer.println(args); // Blocks here
writer.flush();
//Other code
}
public void start(String command) {
try {
ProcessBuilder pb = new ProcessBuilder(command.split(" "));
pb.redirectErrorStream(true);
process = pb.start();
// STDIN of the process.
writer = new PrintWriter(new OutputStreamWriter(process.getOutputStream(), "UTF-8"));
} catch (Exception e) {
e.printStackTrace();
System.out.println("Process ended catastrophically.");
}
}
}
如果我尝试使用
writer.print(args);
writer.print("\n");
它可以在冻结之前处理更大的字符串,但最终仍会锁定。
有没有缓冲流的方法来解决这个问题? print
是否阻塞了具有足够 space 的进程流?
更新
为了回应一些回答和评论,我提供了更多信息。
- 操作系统是Windows7
- BufferedWriter 减慢了 运行 时间,但并没有阻止它最终阻塞。
- 字符串可能会很长,大到 100,000 个字符
- 流程输入已消耗,但按行消耗,即 Scanner.nextLine();
测试代码
import java.io.IOException;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.TimeoutException;
import ProcessRunner.Wrapper;
public class test {
public static void main(String[] args){
System.out.println("Building...");
Wrapper w = new Wrapper("java echo");
System.out.println("Calling...");
String market = "aaaaaa";
for(int i = 0; i < 1000; i++){
try {
System.out.println(w.call(market, 1000));
} catch (InterruptedException | ExecutionException
| TimeoutException e) {
System.out.println("Timed out");
}
market = market + market;
System.out.println("Size = " + market.length());
}
System.out.println("Stopping...");
try {
w.stop();
} catch (IOException e) {
e.printStackTrace();
System.out.println("Stop failed :(");
}
}
}
测试过程:
您必须先编译此文件,并确保 .class 与测试 .class 文件位于同一文件夹中
import java.util.Scanner;
public class echo {
public static void main(String[] args){
while(true){
Scanner stdIn = new Scanner(System.in);
System.out.println(stdIn.nextLine());
}
}
}
如果对等方读取速度比您写入速度慢,则以任何方式写入任何管道或套接字 java.io
块。
对此您无能为力。
我怀疑这里发生的是外部进程正在写入其标准输出。由于您的 Java 代码没有读取它,它最终会填充外部进程的标准输出(或错误)管道。这会阻止外部进程,这意味着它可以从其输入管道中读取....并且您的 Java 进程冻结。
如果这是问题所在,那么使用缓冲写入器将无法解决问题。您需要读取外部进程输出或将其重定向到文件(例如 Linux 上的“/dev/null”)
我正在 java 中编写一个包装程序,它只是应该通过写入其他进程的标准 in 流来将参数传递给其他进程,并从他们的标准输出流读取响应。但是,当我尝试传入的 String
太大时,PrintWriter.print
只会阻塞。没有错误,只是冻结。对此有好的解决方法吗?
相关代码
public class Wrapper {
PrintWriter writer;
public Wrapper(String command){
start(command);
}
public void call(String args){
writer.println(args); // Blocks here
writer.flush();
//Other code
}
public void start(String command) {
try {
ProcessBuilder pb = new ProcessBuilder(command.split(" "));
pb.redirectErrorStream(true);
process = pb.start();
// STDIN of the process.
writer = new PrintWriter(new OutputStreamWriter(process.getOutputStream(), "UTF-8"));
} catch (Exception e) {
e.printStackTrace();
System.out.println("Process ended catastrophically.");
}
}
}
如果我尝试使用
writer.print(args);
writer.print("\n");
它可以在冻结之前处理更大的字符串,但最终仍会锁定。
有没有缓冲流的方法来解决这个问题? print
是否阻塞了具有足够 space 的进程流?
更新
为了回应一些回答和评论,我提供了更多信息。
- 操作系统是Windows7
- BufferedWriter 减慢了 运行 时间,但并没有阻止它最终阻塞。
- 字符串可能会很长,大到 100,000 个字符
- 流程输入已消耗,但按行消耗,即 Scanner.nextLine();
测试代码
import java.io.IOException;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.TimeoutException;
import ProcessRunner.Wrapper;
public class test {
public static void main(String[] args){
System.out.println("Building...");
Wrapper w = new Wrapper("java echo");
System.out.println("Calling...");
String market = "aaaaaa";
for(int i = 0; i < 1000; i++){
try {
System.out.println(w.call(market, 1000));
} catch (InterruptedException | ExecutionException
| TimeoutException e) {
System.out.println("Timed out");
}
market = market + market;
System.out.println("Size = " + market.length());
}
System.out.println("Stopping...");
try {
w.stop();
} catch (IOException e) {
e.printStackTrace();
System.out.println("Stop failed :(");
}
}
}
测试过程:
您必须先编译此文件,并确保 .class 与测试 .class 文件位于同一文件夹中
import java.util.Scanner;
public class echo {
public static void main(String[] args){
while(true){
Scanner stdIn = new Scanner(System.in);
System.out.println(stdIn.nextLine());
}
}
}
如果对等方读取速度比您写入速度慢,则以任何方式写入任何管道或套接字 java.io
块。
对此您无能为力。
我怀疑这里发生的是外部进程正在写入其标准输出。由于您的 Java 代码没有读取它,它最终会填充外部进程的标准输出(或错误)管道。这会阻止外部进程,这意味着它可以从其输入管道中读取....并且您的 Java 进程冻结。
如果这是问题所在,那么使用缓冲写入器将无法解决问题。您需要读取外部进程输出或将其重定向到文件(例如 Linux 上的“/dev/null”)