QT多线程中,对象信号与槽连接不上的问题



        连接不上的问题有很多种,如信号与槽的参数不匹配,参数为自定义类型等等。今天碰到的一个问题是多线程中,信号与槽一直连接不上。防止忘记,记录一下。

        这个问题的情景是一个QObject的派生类不是在主线程里面创建的,然后这个派生类用默认的连接方式去连接信号,这时是连接不上的。

        问题的根源就在于,异步连接的时候,这个信号是由接收者所在的线程的事件处理机制来处理的,如果接收者所在的线程没有事件处理的话,这个信号就不会被处理。看看代码:

voidQCoreApplication::postEvent(QObject*receiver,QEvent*event, int priority)

{

   if (receiver == 0) {

       qWarning("QCoreApplication::postEvent: Unexpected null receiver");

       delete event;

       return;

   }

 

   QThreadData* volatile *pdata= &receiver->d_func()->threadData;

   QThreadData*data = *pdata;

   if (!data) {

       // postingduring destruction? just delete the event to prevent a leak

       delete event;

       return;

   }

 

   // lock the postevent mutex

   data->postEventList.mutex.lock();

 

   // if object hasmoved to another thread, follow it

   while (data != *pdata) {

       data->postEventList.mutex.unlock();

 

       data =*pdata;

       if (!data) {

           //posting during destruction? just delete the event to prevent a leak

           deleteevent;

           return;

       }

 

       data->postEventList.mutex.lock();

   }

 

   QMutexUnlockerlocker(&data->postEventList.mutex);

 

   // if this is oneof the compressible events, do compression

   if (receiver->d_func()->postedEvents

       && self&& self->compressEvent(event,receiver,&data->postEventList)){

       return;

   }

 

   if (event->type() ==QEvent::DeferredDelete&&data == QThreadData::current()) {

       // rememberthe current running eventloop for DeferredDelete

       // eventsposted in the receiver's thread

       static_cast<QDeferredDeleteEvent *>(event)->level =data->loopLevel;

   }

 

   // delete theevent on exceptions to protect against memory leaks till the event is

   // properly ownedin the postEventList

   QScopedPointer<QEvent>eventDeleter(event);

   data->postEventList.addEvent(QPostEvent(receiver,event, priority));

   eventDeleter.take();

   event->posted =true;

   ++receiver->d_func()->postedEvents;

   data->canWait =false;

   locker.unlock();

 

   QAbstractEventDispatcher*dispatcher =data->eventDispatcher.loadAcquire();

   if (dispatcher)

       dispatcher->wakeUp();

}

从上面可以看出,信号被包装成一个事件加到了接收者所在的线程的事件队列当中去了。

暂时想到的几种解决方法:

  1. 在接收者创建线程中,把接收者移动到主线程中:
    pReceiverObj->moveToThread(QApplication::instance()->thread());
    这样发送信号的时候,就会在主线程事件队列处理中来处理了。

  2. connect的最后一个参数改成Qt::DirectConnection,使用直连接的方式来连接信号与槽。

  3. 启动QThread自己的事件队列处理(exec())。
    如果有什么错误和问题,请各位多多指教。

  • 3
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值