成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

關于 Qt 線程同步實例介紹

移動開發
關于 Qt 線程同步實例介紹,實現同步的一個實例,詳細內容請看本文。

Qt 線程同步實例介紹是本文介紹的內容,在Qt中使用線程,沒有Mfc中那么繁瑣,它提供了QThread線程類,提供了創建一個新的方法。線程通過重載QThread::run()函數來完成其操作的,這一點與Java中的線程類相似。

實現一個簡單的繼承自QThread的用戶線程類,代碼如下。

  1. class Thread : public QThread   
  2. {  
  3. public:  
  4.     Thread();  
  5.     void stop();  
  6. protected:  
  7.     virtual void run();  
  8. private:  
  9.     bool m_stop;  
  10. };  
  11. Thread::Thread()  
  12. {  
  13.     m_stop = false;  
  14. }  
  15. void Thread::stop()  
  16. {  
  17.     m_stop = true;  
  18. }  
  19. void Thread::run()  
  20. {  
  21.     while (!m_stop)  
  22.     {  
  23.         sleep(1);  
  24.         qDebug("vic.MINg!");  
  25.     }  
  26.     qDebug("end!");  

在以上的示例中可以看出,線程的編寫并不難!

啟動線程的時候可以,調用函數QThread::start(),開始Thread線程對象。

停止線程的時候可以,調用函數QThread::terminate(),但是terminate()函數并不會立刻終止線程,該線程何時終止取決于操作系統的調度策略。需要注意的是,terminate()函數過于毒辣,它可能在線程執行的任意一步終止執行,從而產生不可預知的后果(如修改某個重要數據時),另外,它也沒有給線程任何清理現場的機會(如釋放內存和鎖等)。

因此,停止線程可以,如上代碼所示,手寫函數stop(),使其線程柔和的退出。

線程停止后,應調用QThread::wait()函數,它使的線程阻塞等待直到退出或超時。

貌似在Unix或Linux下編譯多線程應用程序還必須在.pro文件中加入如下一行,它告訴qmake使用Qt庫中的線程版本。Windows上,Qt庫默認就是線程的。

CONFIG += thread

介紹完了線程的創建,接下來走入正題了,多線程應用程序的一個最普通的需求就是同步幾個線程。Qt提供了以下幾個類來完成這一點:QMutex、QMutexLocker、QSemphore、QWaitCondition。

當然可能還包含QReadWriteLocker、QReadLocker、QWriteLocker,但線程同步是應用很少,這里只做簡單的講解!

QMutex、QMutexLocker

QMutex類提供了一個保護一段臨界區代碼的方法,他每次只允許一個線程訪問這段臨界區代碼。QMutex::lock()函數用來鎖住互斥量,如果互斥量處于解鎖狀態,當前線程就會立即抓住并鎖定它;否則當前線程就會被阻塞,直到持有這個互斥量的線程對其解鎖。線程調用lock()函數后就會持有這個互斥量直到調用unlock()操作為止。QMutex還提供了一個tryLock()函數,如果互斥量已被鎖定,就立即返回。

現在使用QMutex保護上面的線程類的m_stop布爾變量,雖然沒啥用,但這里的目的只是為了演示下QMutex的用法~~

  1. //thread.h頭文件,添加互斥量對象  
  2. private:  
  3.     ...  
  4.     QMutex mutex;  
  5. };  
  6. void Thread::run()  
  7. {  
  8.     forever {  
  9.         mutex.lock();  
  10.         if (m_stop) {  
  11.             m_stop = false;  
  12.             mutex.unlock();  
  13.             break;  
  14.         }  
  15.         mutex.unlock();  
  16.         qDebug("vic.MINg!");  
  17.     }  
  18.     qDebug("end!");  
  19. }  
  20. void Thread::stop()  
  21. {  
  22.     mutex.lock();  
  23.     m_stop = true;  
  24.     mutex.unlock();  

在這里QMutex能夠完全完成互斥操作,但是有些情況下QMutex類是無法某些特定的互斥操作的,下面舉個例子:

#p#

這里我們把void stop()函數,重新定義下,讓他以布爾形式返回,實際也沒有啥用...只為示例的演示效果~~

  1. bool Thread::stop()  
  2. {  
  3.     m_stop = true;  
  4.     return m_stop;  

現在問題出來了,如果要在stop()函數中使用mutex進行互斥操作,但unlock()操作寫在那里?unlock()操作卻不得不再return之后,從而導致unlock()操作永遠也無法執行...

Qt提供了QMutexLocker類何以簡化互斥量的處理,它在構造函數中接受一個QMutex對象作為參數并將其鎖定,在析構函數中解鎖這個互斥量。

這樣可以像下面這樣重新編寫stop()函數:

  1. bool Thread::stop()  
  2. {  
  3.     QMutexLocker locker(&mutex);  
  4.     m_stop = true;  
  5.     return m_stop;  

QReadWriteLocker、QReadLocker、QWriteLocker

下面是一段對QReadWriteLocker類的對象進行,讀寫鎖的操作,比較簡單,這里也不多做講解了,自己看吧 :)

  1. MyData data;  
  2. QReadWriteLock lock;  
  3. void ReaderThread::run()  
  4. {  
  5.     ...  
  6.     lock.lockForRead();  
  7.     access_data_without_modifying_it(&data);  
  8.     lock.unlock();  
  9.     ...  
  10. }  
  11. void WriterThread::run()  
  12. {  
  13.     ...  
  14.     lock.lockForWrite();  
  15.     modify_data(&data);  
  16.     lock.unlock();  
  17.     ...  

QSemphore

Qt中的信號量是由QSemaphore類提供的,信號量可以理解為互斥量功能的擴展,互斥量只能鎖定一次而信號量可以獲取多次,它可以用來保護一定數量的同種資源。

acquire(n)函數用于獲取n個資源,當沒有足夠的資源時調用者將被阻塞直到有足夠的可用資源。release(n)函數用于釋放n個資源。

QSemaphore類還提供了一個tryAcquire(n)函數,在沒有足夠的資源是該函數會立即返回。

一個典型的信號量應用程序是在兩個線程間傳遞一定數量的數據(DataSize),而這兩個線程使用一定大小(BufferSize)的共享循環緩存。

  1. const int DataSize = 100000;  
  2. const int BufferSize = 4096;  
  3. char buffer[BufferSize]; 

生產者線程向緩存中寫入數據,直到它到達終點,然后在起點重新開始,覆蓋已經存在的數據。消費者線程讀取前者產生的數據。

生產者、消費者實例中對同步的需求有兩處,如果生產者過快的產生數據,將會覆蓋消費者還沒有讀取的數據,如果消費者過快的讀取數據,將越過生產者并且讀取到一些垃圾數據。

解決這個問題的一個有效的方法是使用兩個信號量:

  1. QSemaphore freeSpace(BufferSize);  
  2. QSemaphore usedSpace(0); 

freeSpace信號量控制生產者可以填充數據的緩存部分。usedSpace信號量控制消費者可以讀取的區域。這兩個信號量是互補的。其中freeSpace信號量被初始化為BufferSize(4096),表示程序一開始有BufferSize個緩沖區單元可被填充,而信號量usedSpace被初始化為0,表示程序一開始緩沖區中沒有數據可供讀取。

#p#

對于這個實例,每個字節就看作一個資源,實際應用中常會在更大的單位上進行操作,從而減小使用信號量帶來的開銷。

  1. void Producer::run()  
  2. {  
  3.     for (int i = 0; i < DataSize; ++i) {  
  4.         freeSpace.acquire();  
  5.         buffer[i % BufferSize] = "MING"[uint(rand()) % 4];  
  6.         usedSpace.release();  
  7.     }  

在生產者中,我們從獲取一個“自由的”字節開始。如果緩存被消費者還沒有讀取的數據填滿,acquire()的調用就會阻塞,直到消費者已經開始消耗這些數據為止。一旦我們已經獲取了這個字節,我們就用一些隨機數據("M"、"I"、"N"或"G")填充它并且把這個字節釋放為“使用的”,所以它可以被消費者線程使用。

  1. void Consumer::run()  
  2. {  
  3.     for (int i = 0; i < DataSize; ++i) {  
  4.         usedSpace.acquire();  
  5.         cerr << buffer[i % BufferSize];  
  6.         freeSpace.release();  
  7.     }  
  8.     cerr << endl;  

在消費者中,我們從獲取一個“使用的”字節開始。如果緩存中沒有包含任何可讀的數據,acquire()調用將會阻塞,直到生產者已經產生一些數據。一旦我們已經獲取了這個字節,我們就打印它并且把這個字節釋放為“自由的”,使它可以被生產者使用來再次填充數據。

  1. int main()  
  2. {  
  3.     Producer producer;  
  4.     Consumer consumer;  
  5.     producer.start();  
  6.     consumer.start();  
  7.     producer.wait();  
  8.     consumer.wait();  
  9.     return 0;  

main()函數的功能比較簡單,負責啟動生產者和消費者線程,然后等待其各自執行完畢后自動退出。

QWaitCondition

對生產者和消費者問題的另一個解決方法是使用QWaitCondition,它允許線程在一定條件下喚醒其他線程。其中wakeOne()函數在條件滿足時隨機喚醒一個等待線程,而wakeAll()函數則在條件滿足時喚醒所有等待線程
下面重寫生產者和消費者實例,以QMutex為等待條件,QWaitCondition允許一個線程在一定條件下喚醒其他線程

  1. const int DataSize = 100000;  
  2. const int BufferSize = 4096;  
  3. char buffer[BufferSize];  
  4. QWaitCondition bufferIsNotFull;  
  5. QWaitCondition bufferIsNotEmpty;  
  6. QMutex mutex;  
  7. int usedSpace = 0

在緩存之外,我們聲明了兩個QWaitCondition、一個QMutex和一個存儲了在緩存中有多少個“使用的”字節的變量。

  1. void Producer::run()  
  2. {  
  3.     for (int i = 0; i < DataSize; ++i) {  
  4.         mutex.lock();  
  5.         if (usedSpace == BufferSize)  
  6.             bufferIsNotFull.wait(&mutex);  
  7.         buffer[i % BufferSize] = "MING"[uint(rand()) % 4];  
  8.         ++usedSpace;  
  9.         bufferIsNotEmpty.wakeAll();  
  10.         mutex.unlock();  
  11.     }  

在生產者中,我們從檢查緩存是否充滿開始。如果是充滿的,我們等待“緩存不是充滿的”條件。當這個條件滿足時,我們向緩存寫入一個字節,增加usedSpace,并且在喚醒任何等待這個“緩存不是空白的”條件變為真的線程

for循環中的所有語句需要使用互斥量加以保護,以保護其操作的原子性。

  1. bool wait ( QMutex * mutex, unsigned long time = ULONG_MAX ); 

這個函數做下說明,該函數將互斥量解鎖并在此等待,它有兩個參數,第一個參數為一個鎖定的互斥量,第二個參數為等待時間。如果作為第一個參數的互斥量在調用是不是鎖定的或出現遞歸鎖定的情況,wait()函數將立即返回。

調用wait()操作的線程使得作為參數的互斥量在調用前變為鎖定狀態,然后自身被阻塞變成為等待狀態直到滿足以下條件:

其他線程調用了wakeOne()或者wakeAll()函數,這種情況下將返回"true"值。

第二個參數time超時(以毫秒記時),該參數默認情況是ULONG_MAX,表示永不超時,這種情況下將返回"false"值。

wait()函數返回前會將互斥量參數重新設置為鎖定狀態,從而保證從鎖定狀態到等待狀態的原則性轉換。

  1. void Consumer::run()  
  2. {  
  3.     forever {  
  4.         mutex.lock();  
  5.         if (usedSpace == 0)  
  6.             bufferIsNotEmpty.wait(&mutex);  
  7.         cerr << buffer[i % BufferSize];  
  8.         --usedSpace;  
  9.         bufferIsNotFull.wakeAll();  
  10.         mutex.unlock();  
  11.     }  
  12.     cerr << endl;  

消費者做的和生產者正好相反,他等待“緩存不是空白的”條件并喚醒任何等待“緩存不是充滿的”的條件的線程

main()函數與上面的基本相同,這個不再多說。

在QThread類的靜態函數currentThread(),可以返回當前線程線程ID。在X11環境下,這個ID是一個unsigned long類型的值。

小結:關于 Qt 線程同步實例介紹的內容介紹完了,希望本文對你有所幫助。

責任編輯:zhaolei 來源: 互聯網
相關推薦

2011-06-14 16:45:57

Qt 圖標

2011-06-22 15:50:45

QT 線程

2011-08-29 10:22:48

QtWebkit 模塊HTML文檔

2011-06-14 09:46:11

Qt QThread 線程

2011-08-29 10:34:36

QTQWebKitJavaScript

2011-06-30 17:31:32

Qt 多線程 信號

2011-08-25 15:21:02

Lua字符串

2010-03-15 19:37:00

Java多線程同步

2010-03-17 15:34:09

Java線程同步引用

2009-10-12 13:19:14

VB.NET線程同步

2010-03-18 14:09:20

Java線程同步

2011-06-22 16:18:23

QT 多線程 QSocket

2010-06-11 11:24:24

Mrtg教程

2011-07-01 11:18:50

Qt 多線程

2022-08-17 06:25:19

偽共享多線程

2022-08-18 08:24:19

Mysql數據庫

2011-07-01 10:35:20

QT 多線程 TCP

2011-07-05 14:46:34

2011-06-22 10:12:08

Qt 線程

2011-06-27 16:07:49

Qt Designer
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 99久久亚洲 | 久久久做 | 中文字幕一区二区三区不卡在线 | 一区二区三区在线 | 欧 | 国产日韩欧美在线观看 | 99视频在线免费观看 | 欧美日韩精品在线一区 | 国产亚洲一区二区三区在线观看 | 天天成人综合网 | 国户精品久久久久久久久久久不卡 | 中文成人在线 | 免费看爱爱视频 | 看av网 | 午夜欧美| 欧美成人影院在线 | 天天操人人干 | 在线色网站 | 综合色婷婷| 欧美中文字幕一区二区 | 性国产丰满麻豆videosex | 国产激情视频在线免费观看 | 中文字幕免费观看 | 亚洲视频 欧美视频 | 午夜视频精品 | 国产乱码精品一区二区三区五月婷 | 亚洲天堂影院 | av大片| 国产精品99久久久久久宅男 | 黄视频在线网站 | 欧美精品1区2区3区 免费黄篇 | 欧美一区二区三区精品免费 | 欧美在线视频一区二区 | 香蕉视频91| 午夜精品视频在线观看 | 无人区国产成人久久三区 | 欧洲av一区 | 亚洲一区视频 | 国产日韩欧美激情 | 亚洲免费人成在线视频观看 | 日韩在线视频免费观看 | 91精品国产91久久综合桃花 |