Версия для печати темы
Нажмите сюда для просмотра этой темы в оригинальном формате
Форум программистов > С/С++: Кроссплатформенное программирование, Qt/Gtk+/wxWidgets > [Qt]Помогите разобраться с QThread


Автор: nullpoint 25.6.2015, 10:22
Добрый день. Добрался до многопоточного программирования и никак не могу разобраться. 
Например, есть стандартное графическое приложение MainWindow. На форме есть поля для ввода двух матриц, поля для вывода ответа и кнопка, при нажатии на которую запускаются действия с матрицами. Каким образом вынести перемножение матриц и их сложение в параллельные потоки? Функции сложения и перемножения реализованы в одном классе. Не писать же для каждой функции отдельный класс и помещать объект этих классов в отдельный поток.

Автор: math64 25.6.2015, 11:37
В данном случае можно воспользоваться модулем QtConcurent, класс QFutureWatcher<>.
Вместе с Qt идёт пример по паралелному открытию нескольких изображений.
Аналочичным способом можно паралельно складывать и перемножать матрицы.
При использовании QtConcurent автоматически определяется оптимальное количество потоков и расределяются задания между ними.

Автор: nullpoint 26.6.2015, 13:13
math64, спасибо за ответ. Перемножение матриц я привел просто для примера. Меня в общем случае интересует как выполнять методы реализованные в одном классе в параллельных потоках. Во всех найденных примерах, в классе реализован один ключевой метод, который нужно выполнить в отдельном потоке. А как быть с несколькими методами?

Автор: math64 29.6.2015, 10:55
Если требуются разные задания (но не очень много) - используй QThread.
Если требуется множество однотипных заданий - используй QtConcurent. Нужные QThread будут созданы автоматически и задания будут между ними распределены.
Если нужно выпольнить множество заданий нескольких типов, можно использовать switch.
Если выполнение одного задания зависит от другого - QtConcurent не подходит, придётся программировать вручную с помощью QThread.

Автор: nullpoint 29.6.2015, 11:32
math64, можете объяснить поглубже? Например, есть такой класс:
Код

class MyClass{
     /*
         ....
     */
      void method1();
      void method2();
      /*
         ....
     */
}

int main(int argc, char **argv)
{
   //...
   QThread *thread1, *thread2;
}

Как мне выполнить method1() и method2() в разных потоках с помощью QThread?

Автор: math64 29.6.2015, 14:34
1.
Код

class MyClass : public QObject {
Q_OBJECT
...
public slots:
      void method1();
      void method2();
};
class MyThread : public QThread {
Q_OBJECT
public:
...
void run() { emit signalRun(); }
signals:
void signalRun();
};

int main(int argc, char **argv)
{
   MyClass* myClass = new MyClass;
   MyThread *thread1 = new MyThread(), *thread2 = new MyThread;
   connect(thread1, SIGNAL(signalRun()), myClass, SLOT(method1()), Qt::DirectConnection);
   connect(thread2, SIGNAL(signalRun()), myClass, SLOT(method2()), Qt::DirectConnection);
   thread1->start();
   thread2->start();
}

2.
Код

class MyClass{
...
      void method1();
      void method2();
};
class MyThread1: public QThread {
Q_OBJECT
public:
...
void run() { myClass->method1(); }
private:
MyClass* myClass;
};

class MyThread2 : public QThread {
Q_OBJECT
public:
...
void run() { myClass->method2(); }
private:
MyClass* myClass;
};

int main(int argc, char **argv)
{
   MyClass* myClass = new MyClass;
   MyThread1 *thread1 = new MyThread1(myClass);
   MyThread2 *thread2 = new MyThread2(myClass);
   thread1->start();
   thread2->start();
}

Автор: nullpoint 30.6.2015, 09:43
math64, спасибо за примеры кода, но это все же несколько не то, про что я говорил. Тут опять же создается класс наследуемый от QThread, а сами разработчики Qt говорят, что это не совсем правильный путь. Меня интересует, как не создавая собственного класса потоков, выполнять функции в разных потоках. Т.е. что-то типа такого я хочу получить без наследования класса QThread:
Код

class MyClass{
     ...
      void method1();
      void method2();
     ...
}

int main(int argc, char **argv)
{
   ...
   QThread *thread1, *thread2;
   ...
   thread1->start(); //здесь выполняется method1()
   thread2->start(); //здесь выполняется method2()
}

Автор: kuzulis 30.6.2015, 10:00
Цитата

Меня интересует, как не создавая собственного класса потоков, выполнять функции в разных потоках. 


Никак. Весь код (функции и прочее), которые должны выполняться в другом потоке должны быть внутри QThread::run().. Т.е. в любом случае нужно создавать наследника от QThread.

Есть вариант перемещения *всего* объекта в поток, тогда все функции объекта будут выполняться в том потоке, в который объект был перемещен. Для этого не нужно наследоваться от QThread.

Для данной цели (запуска функций в потоках) уже советовали выше использовать QFutureWatcher<>.

PS: И вообще, для этого есть примеры и документация.. Читайте - изучайте - пробуйте.


Автор: nullpoint 30.6.2015, 10:39
kuzulis, так вот я и изучаю документацию. Вопрос и возник из-за того, что в примерах используется наследование от QThread. Хотел узнать возможно ли без этого. Спасибо за ответ.

Автор: math64 30.6.2015, 13:26
Использование moveToThread() мне представляется так:
Код

class MyClass : public QObject {
Q_OBJECT
...
public:
     void emitSignal() { emit mySignal(); }
signals:
     void mySygnal();
public slots:
      void method1();
      void method2();
};
int main(int argc, char **argv)
{
   MyClass* myClass1 = new MyClass;
   MyClass* myClass2 = new MyClass;
   QThread *thread1 = new QThread(), *thread2 = new QThread;
   myClass1->moveToThread(thread1);
   myClass2->moveToThread(thread2);
   connect(myClass1, SIGNAL(mySignal()), myClass1, SLOT(method1()), Qt::QueuedConnection);
   connect(myClass2, SIGNAL(mySignal()), myClass2, SLOT(method2()), Qt::QueuedConnection);
   thread1->start();
   thread2->start();
   myClass1->emitSignal();
   myClass2->emitSignal();
}

Но я не пробовал - может не заработает.
Если экземпляр myClass должен быть один, нужен вспомогательный класс, который из слота вызовет нужный метод - method1() или method2(). Но тогда проще унаследоваться от QThread.

Powered by Invision Power Board (http://www.invisionboard.com)
© Invision Power Services (http://www.invisionpower.com)