亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Qt實現(xiàn)線程與定時器的方法

 更新時間:2022年01月14日 10:11:13   作者:去冰三分糖  
本文主要介紹了Qt實現(xiàn)線程與定時器的方法,文中通過示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下

一、定時器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類定義一個定時器,它可以不停重復(fù),也可以只進行一次便停止。

使用起來也很簡單:

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

創(chuàng)建一個QTimer對象,將信號timeout()與相應(yīng)的槽函數(shù)相連,然后調(diào)用start()函數(shù)。接下來,每隔一段時間,定時器便會發(fā)出一次timeout()信號。

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

二、在多線程中使用QTimer

1.錯誤用法

您可能會這么做:

子類化QThread,在線程類中定義一個定時器,然后在run()方法中調(diào)用定時器的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("當(dāng)前線程id:") << QThread::currentThread();
}

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

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

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

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

——定時器不能被其它線程start。

我們來分析一下:

剛開始只有主線程一個,TestThread的實例是在主線程中創(chuàng)建的,定時器在TestThread的構(gòu)造函數(shù)中,所以也是在主線程中創(chuàng)建的。

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

創(chuàng)建和調(diào)用并不是在同一線程中,所以出現(xiàn)了錯誤。

具體的原理可參考官方文檔——點我

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

默認情況下,QObject處于創(chuàng)建它的線程中。

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

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

2.正確用法一

在TestThread線程啟動后創(chuàng)建定時器。

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);

否則會出現(xiàn)以下警告:

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對象是在主線程中創(chuàng)建的,它的QObject子對象也必須在主線程中創(chuàng)建。所以不能指定父對象為TestThread。

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

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

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

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

上面的代碼將出現(xiàn)以下錯誤:

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()方法改變定時器所處的線程,不要給定時器設(shè)置父類,否則該函數(shù)將不會生效。

在信號槽連接時,我們增加了一個參數(shù)——連接類型,先看看該參數(shù)可以有哪些值:

  • Qt::AutoConnection:默認值。如果接收者處于發(fā)出信號的線程中,則使用Qt::DirectConnection,否則使用Qt::QueuedConnection,連接類型由發(fā)出的信號決定。
  • Qt::DirectConnection:信號發(fā)出后立即調(diào)用槽函數(shù),槽函數(shù)在發(fā)出信號的線程中執(zhí)行。
  • Qt::QueuedConnection:當(dāng)控制權(quán)返還給接收者信號的事件循環(huán)中時,開始調(diào)用槽函數(shù)。槽函數(shù)在接收者的線程中執(zhí)行。

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

到此這篇關(guān)于Qt實現(xiàn)線程與定時器的方法的文章就介紹到這了,更多相關(guān)Qt 線程與定時器 內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • 關(guān)于C/C++中的side effect(負效應(yīng))和sequence point(序列點)

    關(guān)于C/C++中的side effect(負效應(yīng))和sequence point(序列點)

    不知你在寫code時是否遇到這樣的問題?int i = 3; int x = (++i) + (++i) + (++i); 問x值為多少?進行各種理論分析,并在編譯器上實踐,然而可能發(fā)現(xiàn)最終的結(jié)果是不正確的,也是不穩(wěn)定的,不同的編譯器可能會產(chǎn)生不同的結(jié)果。這讓人很頭疼
    2013-10-10
  • C語言實現(xiàn)十六進制與二進制的相互轉(zhuǎn)換

    C語言實現(xiàn)十六進制與二進制的相互轉(zhuǎn)換

    這篇文章主要為大家詳細介紹了如何利用c語言實現(xiàn)將文件中十六進制數(shù)據(jù)與二進制數(shù)據(jù)相互轉(zhuǎn)換,文中的示例代碼講解詳細,具有一定的借鑒價值,感興趣的可以學(xué)習(xí)一下
    2022-11-11
  • 基于堆的基本操作的介紹

    基于堆的基本操作的介紹

    本篇文章對堆的基本操作進行了詳細的分析介紹。需要的朋友參考下
    2013-05-05
  • 適合新手小白DEV?C++的使用方法

    適合新手小白DEV?C++的使用方法

    Dev-C++是一個Windows環(huán)境下C/C++的集成開發(fā)環(huán)境(IDE),它是一款自由軟件,遵守GPL,下面這篇文章主要給大家介紹了關(guān)于適合新手小白DEV?C++的使用方法,文中通過圖文介紹的非常詳細,需要的朋友可以參考下
    2023-02-02
  • C++實現(xiàn)通訊錄系統(tǒng)項目實戰(zhàn)

    C++實現(xiàn)通訊錄系統(tǒng)項目實戰(zhàn)

    這篇文章主要為大家詳細介紹了C++實現(xiàn)通訊錄系統(tǒng)項目實戰(zhàn),文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2022-06-06
  • C++之CNoTrackObject類和new delete操作符的重載實例

    C++之CNoTrackObject類和new delete操作符的重載實例

    這篇文章主要介紹了C++之CNoTrackObject類和new delete操作符的重載實例,是C++程序設(shè)計中比較重要的概念,需要的朋友可以參考下
    2014-10-10
  • c語言中字符串與字符串?dāng)?shù)組詳解

    c語言中字符串與字符串?dāng)?shù)組詳解

    在C語言當(dāng)中,字符串?dāng)?shù)組可以使用char a[] [10]; 或者char *a[]; 表示,下面這篇文章主要給大家介紹了關(guān)于c語言中字符串與字符串?dāng)?shù)組的相關(guān)資料,需要的朋友可以參考下
    2021-11-11
  • QT網(wǎng)絡(luò)通信TCP客戶端實現(xiàn)詳解

    QT網(wǎng)絡(luò)通信TCP客戶端實現(xiàn)詳解

    這篇文章主要為大家詳細介紹了QT網(wǎng)絡(luò)通信TCP客戶端實現(xiàn),文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2022-08-08
  • c++中nlohmann?json的基本使用教程

    c++中nlohmann?json的基本使用教程

    nlohmann/json 是一個C++實現(xiàn)的JSON解析器,使用非常方便直觀,下面這篇文章主要給大家介紹了關(guān)于c++中nlohmann?json基本使用的相關(guān)資料,文中通過實例代碼介紹的非常詳細,需要的朋友可以參考下
    2022-09-09
  • C語言代碼實現(xiàn)三子棋游戲

    C語言代碼實現(xiàn)三子棋游戲

    這篇文章主要為大家詳細介紹了C語言代碼實現(xiàn)三子棋游戲,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2020-11-11

最新評論