QT通過(guò)三種形式提供了對(duì)線程的支持。它們分別是,,一,、平臺(tái)無(wú)關(guān)的線程類,二,、線程安全的事件投遞,,三、跨線程的信號(hào)-槽連接,。這使得開(kāi)發(fā)輕巧的多線程Qt程序更為容易,,并能充分利用多處理器機(jī)器的優(yōu)勢(shì)。多線程編程也是一個(gè)有用的模式,,它用于解決執(zhí)行較長(zhǎng)時(shí)間的操作而不至于用戶界面失去響應(yīng),。在Qt的早期版本中,在構(gòu)建庫(kù)時(shí)有不選擇線程支持的選項(xiàng),,從4.0開(kāi)始,,線程總是有效的。
線程類 Qt 包含下面一些線程相關(guān)的類: QThread 提供了開(kāi)始一個(gè)新線程的方法 QThreadStorage 提供逐線程數(shù)據(jù)存儲(chǔ) QMutex 提供相互排斥的鎖,,或互斥量 QMutexLocker 是一個(gè)便利類,,它可以自動(dòng)對(duì)QMutex加鎖與解鎖 QReadWriterLock 提供了一個(gè)可以同時(shí)讀操作的鎖 QReadLocker與QWriteLocker 是便利類,它自動(dòng)對(duì)QReadWriteLock加鎖與解鎖 QSemaphore 提供了一個(gè)整型信號(hào)量,,是互斥量的泛化 QWaitCondition 提供了一種方法,,使得線程可以在被另外線程喚醒之前一直休眠。 創(chuàng)建一個(gè)線程 為創(chuàng)建一個(gè)線程,,子類化QThread并且重寫(xiě)它的run()函數(shù),,例如: class MyThread : public QThread { Q_OBJECT protected: void MyThread::run() 線程同步 void WriterThread::run() QSemaphore 是QMutex的一般化,,它可以保護(hù)一定數(shù)量的相同資源,,與此相對(duì),一個(gè)mutex只保護(hù)一個(gè)資源,。下面例子中,,使用QSemaphore來(lái)控制對(duì)環(huán)狀緩沖的訪問(wèn),此緩沖區(qū)被生產(chǎn)者線程和消費(fèi)者線程共享,。生產(chǎn)者不斷向緩沖寫(xiě)入數(shù)據(jù)直到緩沖末端,,再?gòu)念^開(kāi)始。消費(fèi)者從緩沖不斷讀取數(shù)據(jù),。信號(hào)量比互斥量有更好的并發(fā)性,,假如我們用互斥量來(lái)控制對(duì)緩沖的訪問(wèn),那么生產(chǎn)者,,消費(fèi)者不能同時(shí)訪問(wèn)緩沖,。然而,,我們知道在同一時(shí)刻,不同線程訪問(wèn)緩沖的不同部分并沒(méi)有什么危害,。 const int DataSize = 100000; QSemaphore freeBytes(BufferSize); class Producer : public QThread void Producer::run() class Consumer : public QThread void Consumer::run() int main(int argc, char *argv[]) 下面的例子中,生產(chǎn)者首先必須檢查緩沖是否已滿(numUsedBytes==BufferSize),,如果是,,線程停下來(lái)等待bufferNotFull條件。如果不是,,在緩沖中生產(chǎn)數(shù)據(jù),,增加numUsedBytes,激活條件 bufferNotEmpty。使用mutex來(lái)保護(hù)對(duì)numUsedBytes的訪問(wèn),。另外,,QWaitCondition::wait()接收一個(gè)mutex作為參數(shù),這個(gè)mutex應(yīng)該被調(diào)用線程初始化為鎖定狀態(tài),。在線程進(jìn)入休眠狀態(tài)之前,,mutex會(huì)被解鎖。而當(dāng)線程被喚醒時(shí),,mutex會(huì)處于鎖定狀態(tài),而且,,從鎖定狀態(tài)到等待狀態(tài)的轉(zhuǎn)換是原子操作,這阻止了競(jìng)爭(zhēng)條件的產(chǎn)生,。當(dāng)程序開(kāi)始運(yùn)行時(shí),,只有生產(chǎn)者可以工作。消費(fèi)者被阻塞等待bufferNotEmpty條件,,一旦生產(chǎn)者在緩沖中放入一個(gè)字節(jié),,bufferNotEmpty條件被激發(fā),消費(fèi)者線程于是被喚醒,。 const int DataSize = 100000; QWaitCondition bufferNotEmpty; class Producer : public QThread void Producer::run() for (int i = 0; i < DataSize; ++i) { buffer[i % BufferSize] = "ACGT"[(int)qrand() % 4]; mutex.lock(); class Consumer : public QThread void Consumer::run() fprintf(stderr, "%c", buffer[i % BufferSize]); mutex.lock(); int main(int argc, char *argv[]) |
|
來(lái)自: 昵稱4765400 > 《我的圖書(shū)館》