91超碰碰碰碰久久久久久综合_超碰av人澡人澡人澡人澡人掠_国产黄大片在线观看画质优化_txt小说免费全本

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

如何使用Qt實現線程與定時器

發布時間:2022-01-14 11:23:53 來源:億速云 閱讀:494 作者:小新 欄目:開發技術

這篇文章主要介紹如何使用Qt實現線程與定時器,文中介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們一定要看完!

一、定時器QTimer類

The QTimer class provides repetitive and single-shot timers.

The QTimer class provides a high-level programming interface for timers. To use it, create a QTimer, connect its timeout() signal to the appropriate slots, and call start(). From then on, it will emit the timeout() signal at constant intervals.

上面這段話摘自Qt助手文檔,我們使用QTimer類定義一個定時器,它可以不停重復,也可以只進行一次便停止。

使用起來也很簡單:

QTimer *timer = new QTimer(this);
connect(timer, SIGNAL(timeout()), this, SLOT(update()));
timer->start(1000);

創建一個QTimer對象,將信號timeout()與相應的槽函數相連,然后調用start()函數。接下來,每隔一段時間,定時器便會發出一次timeout()信號。

更多用法這里就不講了,您可以自行參考官方文檔。比如如何停止、如何令定時器只運行一次等。

二、在多線程中使用QTimer

1.錯誤用法

您可能會這么做:

子類化QThread,在線程類中定義一個定時器,然后在run()方法中調用定時器的start()方法。

TestThread::TestThread(QObject *parent)
    : QThread(parent)
{
    m_pTimer = new QTimer(this);
    connect(m_pTimer, &QTimer::timeout, this, &TestThread::timeoutSlot);
}
 
void TestThread::run()
{
    m_pTimer->start(1000);
}
 
void TestThread::timeoutSlot()
{
    qDebug() << QString::fromLocal8Bit("當前線程id:") << QThread::currentThread();
}

接下來在主線程中創建該線程對象,并調用它的start()方法:

m_pThread = new TestThread(this);
m_pThread->start();

看似十分自然,沒有什么不妥,然而,編譯器將通知下面的錯誤信息:

 QObject::startTimer: Timers cannot be started from another thread 

&mdash;&mdash;定時器不能被其它線程start。

我們來分析一下:

剛開始只有主線程一個,TestThread的實例是在主線程中創建的,定時器在TestThread的構造函數中,所以也是在主線程中創建的。

當調用TestThread的start()方法時,這時有兩個線程。定時器的start()方法是在另一個線程中,也就是TestThread中調用的。

創建和調用并不是在同一線程中,所以出現了錯誤。

具體的原理可參考官方文檔&mdash;&mdash;點我

每個QObject實例都有一個叫做“線程關系”(thread affinity)的屬性,或者說,它處于某個線程中。

默認情況下,QObject處于創建它的線程中。

當QObject接收隊列信號(queued signal)或者傳來的事件(posted event),槽函數或事件處理器將在對象所處的線程中執行。

根據以上的原理,Qt使用計時器的線程關系(thread affinity)來決定由哪個線程發出timeout()信號。正因如此,你必須在它所處的線程中start或stop該定時器,在其它線程中啟動定時器是不可能的。

2.正確用法一

在TestThread線程啟動后創建定時器。

void TestThread::run()
{
    m_pTimer = new QTimer();
    m_pTimer->setInterval(1000);
    connect(m_pTimer, &QTimer::timeout, this, &TestThread::timeoutSlot);
    m_pTimer->start();
    this->exec();
}

有些地方需要注意:

1.不能像下面這樣給定時器指定父對象

m_pTimer = new QTimer(this);

否則會出現以下警告:

QObject: Cannot create children for a parent that is in a different thread.
(Parent is TestThread(0x709d88), parent's thread is QThread(0x6e8be8), current thread is TestThread(0x709d88) 

因為TestThread對象是在主線程中創建的,它的QObject子對象也必須在主線程中創建。所以不能指定父對象為TestThread。

2.必須要加上事件循環exec()

否則線程會立即結束,并發出finished()信號。

另外還有一點需要注意,與start一樣,定時器的stop也必須在TestThread線程中,否則會出錯。

void TestThread::timeoutSlot()
{
    m_pTimer->stop();
    qDebug() << QString::fromLocal8Bit("當前線程id:") << QThread::currentThread();
}

上面的代碼將出現以下錯誤:

QObject::killTimer: Timers cannot be stopped from another thread

綜上,子類化線程類的方法可行,但是不太好。 

3.正確用法二

無需子類化線程類,通過信號啟動定時器。

TestClass::TestClass(QWidget *parent)
    : QWidget(parent)
{
    m_pThread = new QThread(this);
    m_pTimer = new QTimer();
    m_pTimer->moveToThread(m_pThread);
    m_pTimer->setInterval(1000);
    connect(m_pThread, SIGNAL(started()), m_pTimer, SLOT(start()));
    connect(m_pTimer, &QTimer::timeout, this, &ThreadTest::timeOutSlot, Qt::DirectConnection);
}

通過moveToThread()方法改變定時器所處的線程,不要給定時器設置父類,否則該函數將不會生效。

在信號槽連接時,我們增加了一個參數&mdash;&mdash;連接類型,先看看該參數可以有哪些值:

  • Qt::AutoConnection:默認值。如果接收者處于發出信號的線程中,則使用Qt::DirectConnection,否則使用Qt::QueuedConnection,連接類型由發出的信號決定。

  • Qt::DirectConnection:信號發出后立即調用槽函數,槽函數在發出信號的線程中執行。

  • Qt::QueuedConnection:當控制權返還給接收者信號的事件循環中時,開始調用槽函數。槽函數在接收者的線程中執行。

回到我們的例子,首先將定時器所處的線程改為新建的線程,然后連接信號槽,槽函數在定時器所處的線程中執行。

以上是“如何使用Qt實現線程與定時器”這篇文章的所有內容,感謝各位的閱讀!希望分享的內容對大家有幫助,更多相關知識,歡迎關注億速云行業資訊頻道!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

qt
AI

陵川县| 云南省| 岳普湖县| 南皮县| 杭锦后旗| 马尔康县| 永济市| 嘉义县| 新兴县| 东至县| 祁连县| 信丰县| 南开区| 江油市| 文水县| 罗山县| 盘锦市| 来安县| 屏南县| 德州市| 内乡县| 中牟县| 临清市| 古蔺县| 民权县| 扶沟县| 梁山县| 株洲县| 宿松县| 克山县| 永仁县| 石狮市| 阳城县| 改则县| 凭祥市| 临安市| 合作市| 高密市| 正蓝旗| 乐至县| 闵行区|