Помощь - Поиск - Пользователи - Календарь
Полная версия этой страницы: Помогите пожалуйста наладить перехват stdout консольного приложения.
Форум на CrossPlatform.RU > Библиотеки > Qt > Qt Ввод/Вывод, Сеть. Межпроцессное взаимодействие
Gordey
Всем привет.
Помогите пожалуйста наладить перехват stdout консольного приложения.

Использую QT 4.6.3 и Microsoft Visual Studio 2008.

Задача: из Qt GUI запустить консольное приложение, перехватить его стандартный поток ввода/вывода и вывести перехваченные данные в элемент QTextEdit.

Консольное приложение в цикле валит в поток вывода строки
Value: 1
Value: 2
Value: 3
и т.д.

#include <iostream>

int main(int argc, char* argv[])
{
    for(int i =0; i != 1000000; ++i)
        std::cout << "Value: " << i << std::endl;

    return 0;
}


В Qt GUI при нажатии на кнопку создается отдельный поток.

process = new QProcess;
process->setProcessChannelMode(QProcess::MergedChannels);
connect(process, SIGNAL(readyReadStandardOutput()), this, SLOT(output()));
process->start("test_console.exe");
exec();

При создании потока запускаем процесс с консольным приложением. При получении сигнала readyReadStandardOutput() отправляем считанные данные родительскому приложению.

void ServerThread::output()
{
    QByteArray bytes = process->readAllStandardOutput();
    emit updateOutput(bytes);
}

В Qt GUI получаем посылку с данными и выводим ее в QTextEdit.

void MainWindow::setOutput(const QString &msg)
{
    ui->textEdit->insertPlainText(msg);
}

Проблема в том, что при перехвате потока ввода/вывода и выводе данных в QTextEdit происходит залипание Qt GUI. То есть я практически ничего не могу сделать с родительским окном.
Как можно решить данную проблему? Как избавиться от залипания QT окна? При решении проблемы изменять код консольного приложения нельзя , нужно решить задачу другими способами.

PS: Прикладываю тестовые проекты приложений. Так как проекты тестовые строго не судите за код… Обработки завершения процессов и потоков я не делал. Для распаковки архива с примерами удалите ".txt".
Gordey
Дополнительно выложил файлы проектов на webfile.ru

Скачать test.rar с WebFile.RU
Litkevich Yuriy
читай по форуму темы типа "потоки и сигналы", чтобы разобраться в каком конкретно потоке исполняется слот и т.п.

П.С. вопрос не в первый раз поднимается.
Gordey
Цитата(Litkevich Yuriy @ 1.7.2010, 22:44) *
читай по форуму темы типа "потоки и сигналы", чтобы разобраться в каком конкретно потоке исполняется слот и т.п.

П.С. вопрос не в первый раз поднимается.


Да вроде бы все перерыл. Отчаявшись решил вот у гуру спросить.
Если ты знаешь решение проблемы уж подскажи пожалуйста, а не отсылай в далекие дали. Я там уже был, но не встретил того что мне нужно.

Litkevich Yuriy
яб для начала, отправлял бы всё в отладочную консоль (с помощью qDebug), вместо отображения в виджете. тут можно было бы сразу сказать тормозит ли виджет от потока данных или всё таки проблемы с потоками выполнения кода.
Gordey
Цитата(Litkevich Yuriy @ 1.7.2010, 23:28) *
яб для начала, отправлял бы всё в отладочную консоль (с помощью qDebug), вместо отображения в виджете. тут можно было бы сразу сказать тормозит ли виджет от потока данных или всё таки проблемы с потоками выполнения кода.


Если я комментирую вывод информации в виджет, то тормоза пропадают.

void MainWindow::setOutput(const QString &msg)
{
// ui->plainTextEdit->insertPlainText(msg);
}


Залипание можно убрать если после вывода строки поставить задержку Sleep(1).

Старый вариант:
int main(int argc, char* argv[])
{
for(int i =0; i != 1000000; ++i)
std::cout << "Value: " << i << std::endl;

return 0;
}

Новый вариант:
int main(int argc, char* argv[])
{
for(int i =0; i != 1000000; ++i)
{
std::cout << "Value: " << i << std::endl;
Sleep(1);
}

return 0;
}

Но такой вариант мне не подходит так как консольное приложения я изменить не могу.
Litkevich Yuriy
Цитата(Gordey @ 2.7.2010, 10:08) *
Но такой вариант мне не подходит так как консольное приложения я изменить не могу.
а зачем ты в него вообще полез, если оно неизменяемое? Про него и забудь.

Видимо для TextEdit это слишком большой темп поступления данных. Можно попробовать буферировать его, хотя я сомневаюсь, что это поможет. Скорее всего придётся менять сам виджет.
Gordey
Цитата(Litkevich Yuriy @ 2.7.2010, 12:50) *
Видимо для TextEdit это слишком большой темп поступления данных. Можно попробовать буферировать его, хотя я сомневаюсь, что это поможет. Скорее всего придётся менять сам виджет


А что значит менять? Как его можно изменить чтобы он быстрее работал?
Litkevich Yuriy
Цитата(Gordey @ 2.7.2010, 19:31) *
А что значит менять?
на какой-нибудь другой виджет.
Gordey
Цитата(Litkevich Yuriy @ 2.7.2010, 17:22) *
TextEdit


Ааа... Я уже разные перепробовал.
Gordey
Неужто здесь на форуме нет гуру способных помочь?
Алексей1153
Цитата(Gordey @ 2.7.2010, 9:08) *
Если я комментирую вывод информации в виджет, то тормоза пропадают.

void MainWindow::setOutput(const QString &msg)
{
// ui->plainTextEdit->insertPlainText(msg);
}

а если накапливать данные в std::vector<char> , а закидывать их с контрол не чаще чем, скажем, раз в 250 мс ? Причём, если очередное обновление запоздало, то основной процесс пускай подпинывает обновление контрола из вектора по таймеру
inviZ
Gordey, может попробуешь QPlainTextEdit? Он попроще. Все-таки QTextEdit достаточно сложный с кучей плюшек всяких и поэтому тормозной.
Gordey
Цитата(inviZ @ 4.7.2010, 14:13) *
Gordey, может попробуешь QPlainTextEdit? Он попроще. Все-таки QTextEdit достаточно сложный с кучей плюшек всяких и поэтому тормозной.


Пробовал уже. На первый взгляд никаких изменений не увидел.

Цитата(Алексей1153 @ 4.7.2010, 10:34) *
Цитата(Gordey @ 2.7.2010, 9:08) *
Если я комментирую вывод информации в виджет, то тормоза пропадают.

void MainWindow::setOutput(const QString &msg)
{
// ui->plainTextEdit->insertPlainText(msg);
}

а если накапливать данные в std::vector<char> , а закидывать их с контрол не чаще чем, скажем, раз в 250 мс ? Причём, если очередное обновление запоздало, то основной процесс пускай подпинывает обновление контрола из вектора по таймеру


И так уже пробовал. Не помогает. Посылки приходят уже большого размера.
Litkevich Yuriy
я склоняюсь к тому, что при большом темпе поступления данных, в Qt будут проблемы. Т.к. вопрос о медленной отрисовке шрифтов в Qt уже не однократно поднимался.

Могу ещё для эксперимента несколько идей подкинуть:
1) обрамлять вставку данных в виджет функцией setUpdatesEnabled(bool)
2) задать виджету фиксированный размер, чтобы не заставлять компоновщик обсчитывать геометрию.
Gordey
Цитата(Litkevich Yuriy @ 4.7.2010, 15:55) *
я склоняюсь к тому, что при большом темпе поступления данных, в Qt будут проблемы. Т.к. вопрос о медленной отрисовке шрифтов в Qt уже не однократно поднимался.

Могу ещё для эксперимента несколько идей подкинуть:
1) обрамлять вставку данных в виджет функцией setUpdatesEnabled(bool)
2) задать виджету фиксированный размер, чтобы не заставлять компоновщик обсчитывать геометрию.


Спасибо за идеи. Попробую.
Для просмотра полной версии этой страницы, пожалуйста, пройдите по ссылке.
Форум IP.Board © 2001-2024 IPS, Inc.