F
14 февраля 2018 г. 23:06

компонент класса, созданный в куче после переноса самого обьекта в каком потоке окажется?

qt, QThread, moveToThread, GUI поток

Привет ВСем! Итак вопрос такой: Есть класс, в полях которого есть указатель на объект наследник от QDialog. Я создаю обьект основного класса, в конструкторе этого класса я создаю обьект в динамической памяти - указатель на наследника QDialog. Обьеект оснновного класса далее я перевожу из GUI потока в другой. Обьект в поле основного класса который я создал динамически в его конструкторе, остается в GUI потоке. Почему при вызове exec у наследника QDialog создается форма диалоога и все виснет?

Base.h
class Base
{
  public:
  Base();

  private:
  QDeriveDialog* m_DeriveDialog;

}
Base.cpp:
Base::Base
{
m_DeriveDialog = new QDeriveDialog();
}
...
в основной программе:
qDebug() << thread();      // запомним поток GUI
QThread thr = new QThread;
Base* base = new Base;
base->moveTothread(thr);
...
Base.cpp:
void process()
{
   qDebug() << m_DerivedDialog->thread;   // проверяю к какому потоку принадлежит объект
                                          // и он равен потоку GUI,
   if(m_DerivedDialog->exec() == QDialog::Rejected) // однако при вызове диал.окна, в нем 
// все виснет, окно появляется только контуром,
// все остальное не прорисовывается и курсор в
// колесико превращается ret = false; ... }
3

Вам это нравится? Поделитесь в социальных сетях!

3
Evgenii Legotckoi
  • 15 февраля 2018 г. 3:34

Добрый день!


Я же вам уже говорил, что нельзя переносить объекты графического интерфейса в другие потоки. Они работают только в главном GUI потоке.
Вот выдержка из официальной документации Qt

As mentioned, each program has one thread when it is started. This thread is called the "main thread" (also known as the "GUI thread" in Qt applications). The Qt GUI must run in this thread. All widgets and several related classes, for example QPixmap don't work in secondary threads.
Смысл данного текста состоит в том, что графические объекты и объекты интерфейса нельзя переносить в другие потоки, то есть использование moveToThread является неправильным.
Вот ссылка на документацию , где это написано
    F
    • 15 февраля 2018 г. 14:47

    В коде я указал, что он не перенёсся.

      Evgenii Legotckoi
      • 15 февраля 2018 г. 14:51
      • (ред.)

      А я вам сказал, что так делать в принципе нельзя. Нечего пытаться делать то, чего делать нельзя и о чём сказано в документации. Я не вижу смысла даже проверять куда перенёсся объект, поскольку в документации сказано, что подобный подход, который вы попытались применить недопустим.

       
      Война с ветряными мельницами - самое бесполезное занятие.

      Нужно работать с данными в других потоках и возвращать результат расчётов в GUI-поток для отображения в виджетах.
       
      Естественно, что виджет диалога не перенёсся, поскольку такое нельзя делать, Вы просто положили программу и всё. Тут даже думать нечего. 
      Попытка перенести виджет в поток, отличный от GUI поток как правило либо не имеет эффекта, либо ложит программу.

      ТАК ДЕЛАТЬ НЕЛЬЗЯ

        Комментарии

        Только авторизованные пользователи могут публиковать комментарии.
        Пожалуйста, авторизуйтесь или зарегистрируйтесь