Java: запуск подпроцесса с собственным терминалом
У меня есть два контроллера и одна программа, которая запускает их обоих. Одним из них является эмулятор, который производит данные, другой анализирует данные. Два контроллера зависят друг от друга и взаимодействуют с помощью RMI. Поэтому я запускаю эмулятор в другом потоке, а анализатор - в основном.Это прекрасно работает.
Теперь проблема в том, что они оба выводят довольно много информации на консоль, и я действительно хотел бы, чтобы они печатали на двух разных терминалах. Есть способ сделать это?
Я попытался запустить эмулятор как подпроцесс в новой командной строке (следующим шагом будет независимость от платформы)
String separator = System.getProperty("file.separator");
String classpath = System.getProperty("java.class.path");
String path = System.getProperty("java.home")
+ separator + "bin" + separator + "java";
ProcessBuilder processBuilder =
new ProcessBuilder(
"\"" + path + "\"",
"-cp",
classpath,
TheEumlator.class.getName());
String command = StringUtils.join(processBuilder.command(), " ");
Process p = Runtime.getRuntime().exec(String.format("cmd /c start cmd.exe /K %s", command));
Тем не менее classpath
это слишком долго, и выход cmd.exe
является The command line is too long.
У вас есть идея, как создать другой поток или процесс с собственным выходным терминалом? Буду рад любому предложению.
ура
ОБНОВИТЬ
Я совместил ответ ОлавиМустаноя с этим решением http://unserializableone.blogspot.co.uk/2009/01/redirecting-systemout-and-systemerr-to.html
Теперь он использует стандартные System.out
, System.err
и трассировка стека. Кроме того, это прокручивает.
public class ConsoleWindow implements Runnable {
private String title;
private JFrame frame;
private JTextArea outputArea;
private JScrollPane scrollPane;
public ConsoleWindow(String title, boolean redirectStreams) {
this.title = title;
this.outputArea = new JTextArea(30, 80);
this.outputArea.setEditable(false);
if (redirectStreams)
redirectSystemStreams();
}
public ConsoleWindow(String title) {
this(title, false);
}
@Override
public void run() {
frame = new JFrame(this.title);
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
scrollPane = new JScrollPane(outputArea);
JPanel outputPanel = new JPanel(new FlowLayout());
outputPanel.add(scrollPane);
frame.add(outputPanel);
frame.pack();
frame.setVisible(true);
}
private void updateTextArea(final String text) {
SwingUtilities.invokeLater(new Runnable() {
public void run() {
outputArea.append(text);
}
});
}
private void redirectSystemStreams() {
OutputStream out = new OutputStream() {
@Override
public void write(int b) throws IOException {
updateTextArea(String.valueOf((char) b));
}
@Override
public void write(byte[] b, int off, int len) throws IOException {
updateTextArea(new String(b, off, len));
}
@Override
public void write(byte[] b) throws IOException {
write(b, 0, b.length);
}
};
System.setOut(new PrintStream(out, true));
System.setErr(new PrintStream(out, true));
}
public void println(String msg) {
updateTextArea(msg + "\n");
}
public void println(Throwable t) {
println(t.toString());
}
public void print(String msg) {
updateTextArea(msg);
}
public void printStackTrace(Throwable t) {
StringWriter sw = new StringWriter();
PrintWriter pw = new PrintWriter(sw);
t.printStackTrace(pw);
this.println(sw.toString());
}
}
2 ответа
Нужно ли выводить сообщения на эти консоли? Если нет, вы можете просто реализовать класс (скажем, OutputWindow), который каким-то образом принимает сообщения и показывает их в окне.
Чтобы лучше проиллюстрировать то, что я высказал в своем комментарии, рассмотрим следующую реализацию OutputWindow
:
class OutputWindow implements Runnable {
private String title;
private JFrame frame;
private JTextArea outputArea;
public OutputWindow(String title) {
this.title = title;
this.outputArea = new JTextArea(15, 50);
this.outputArea.setEditable(false);
}
@Override
public void run() {
frame = new JFrame(this.title);
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
JPanel outputPanel = new JPanel(new FlowLayout());
outputPanel.add(outputArea);
frame.add(outputPanel);
frame.pack();
frame.setVisible(true);
}
public void println(String msg) {
outputArea.append(msg + "\n");
}
public void println(Throwable t) {
this.println(t.toString());
}
public void print(String msg) {
outputArea.append(msg);
}
public void printStackTrace(Throwable t) {
StringWriter sw = new StringWriter();
PrintWriter pw = new PrintWriter(sw);
t.printStackTrace(pw);
this.println(sw.toString());
}
}
Простой, довольно быстро проверенный фрагмент кода. Можно открыть новую "консоль", например так:
OutputWindow console = new OutputWindow("Console");
SwingUtilities.invokeLater(console);
Вы можете попробовать это:
OutputWindow console = new OutputWindow("Console");
SwingUtilities.invokeLater(console);
try {
System.out.println(2 / 0); // just to throw an exception
} catch (ArithmeticException ex) {
console.printStackTrace(ex);
}
Преимущество в этом заключается в том, насколько вы можете настроить его! Безграничные возможности. Создайте свою собственную современную консоль с супер удивительными цветами и любой функциональностью.
Самое простое решение - записать выходные данные в файлы, вручную открыть свои терминалы и запустить хвост блокировки.