• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            Qt 中的多線程 

            QT通過三種形式提供了對線程的支持。它們分別是,一、平臺無關的線程類,二、線程安全的事件投遞,三、跨線程的信號-槽連接。這使得開發輕巧的多線程Qt程序更為容易,并能充分利用多處理器機器的優勢。多線程編程也是一個有用的模式,它用于解決執行較長時間的操作而不至于用戶界面失去響應。在Qt的早期版本中,在構建庫時有不選擇線程支持的選項,從4.0開始,線程總是有效的。

            線程類

            Qt 包含下面一些線程相關的類:
            QThread 提供了開始一個新線程的方法
            QThreadStorage 提供逐線程數據存儲
            QMutex  提供相互排斥的鎖,或互斥量
            QMutexLocker 是一個便利類,它可以自動對QMutex加鎖與解鎖
            QReadWriterLock 提供了一個可以同時讀操作的鎖
            QReadLockerQWriteLocker 是便利類,它自動對QReadWriteLock加鎖與解鎖
            QSemaphore 提供了一個整型信號量,是互斥量的泛化
            QWaitCondition 提供了一種方法,使得線程可以在被另外線程喚醒之前一直休眠。

            創建一個線程

            為創建一個線程,子類化QThread并且重寫它的run()函數,例如:
            class MyThread : public QThread
             {
                 Q_OBJECT

             protected:
                 void run();
             };

             void MyThread::run()
             {
                 ...
             }
            之后,創建這個線程對象的實例,調用QThread::start()。于是,在run()里出現的代碼將會在另外線程中被執行。
            注意:QCoreApplication::exec()必須總是在主線程(執行main()的那個線程)中被調用,不能從一個QThread中調用。在GUI程序中,主線程也被稱為GUI線程,因為它是唯一一個允許執行GUI相關操作的線程。另外,你必須在創建一個QThread之前創建QApplication(or QCoreApplication)對象。

            線程同步

            QMutex, QReadWriteLock, QSemaphoreQWaitCondition 提供了線程同步的手段。使用線程的主要想法是希望它們可以盡可能并發執行,而一些關鍵點上線程之間需要停止或等待。例如,假如兩個線程試圖同時訪問同一個全局變量,結果可能不如所愿。
            QMutex  提供相互排斥的鎖,或互斥量。在一個時刻至多一個線程擁有mutex,假如一個線程試圖訪問已經被鎖定的mutex,那么它將休眠,直到擁有mutex的線程對此mutex解鎖。Mutexes常用來保護共享數據訪問。
             QReadWriterLock 與QMutex相似,除了它對 "read","write"訪問進行區別對待。它使得多個讀者可以共時訪問數據。使用QReadWriteLock而不是QMutex,可以使得多線程程序更具有并發性。

             QReadWriteLock lock;
             void ReaderThread::run()
             {
                // ...
                 lock.lockForRead();
                 read_file();
                 lock.unlock();
                 //...
             }

             void WriterThread::run()
             {
                // ...
                 lock.lockForWrite();
                 write_file();
                 lock.unlock();
                // ...
             }

            QSemaphoreQMutex的一般化,它可以保護一定數量的相同資源,與此相對,一個mutex只保護一個資源。下面例子中,使用QSemaphore來控制對環狀緩沖的訪問,此緩沖區被生產者線程和消費者線程共享。生產者不斷向緩沖寫入數據直到緩沖末端,再從頭開始。消費者從緩沖不斷讀取數據。信號量比互斥量有更好的并發性,假如我們用互斥量來控制對緩沖的訪問,那么生產者,消費者不能同時訪問緩沖。然而,我們知道在同一時刻,不同線程訪問緩沖的不同部分并沒有什么危害。

             const int DataSize = 100000;
             const int BufferSize = 8192;
             char buffer[BufferSize];

             QSemaphore freeBytes(BufferSize);
             QSemaphore usedBytes;

             class Producer : public QThread
             {
             public:
                 void run();
             };

             void Producer::run()
             {
                 qsrand(QTime(0,0,0).secsTo(QTime::currentTime()));
                 for (int i = 0; i < DataSize; ++i) {
                     freeBytes.acquire();
                     buffer[i % BufferSize] = "ACGT"[(int)qrand() % 4];
                     usedBytes.release();
                 }
             }

             class Consumer : public QThread
             {
             public:
                 void run();
             };

             void Consumer::run()
             {
                 for (int i = 0; i < DataSize; ++i) {
                     usedBytes.acquire();
                     fprintf(stderr, "%c", buffer[i % BufferSize]);
                     freeBytes.release();
                 }
                 fprintf(stderr, "\n");
             }

             int main(int argc, char *argv[])
             {
                 QCoreApplication app(argc, argv);
                 Producer producer;
                 Consumer consumer;
                 producer.start();
                 consumer.start();
                 producer.wait();
                 consumer.wait();
                 return 0;
             }
            QWaitCondition 允許線程在某些情況發生時喚醒另外的線程。一個或多個線程可以阻塞等待一QWaitCondition ,用wakeOne()或wakeAll()設置一個條件。wakeOne()隨機喚醒一個,wakeAll()喚醒所有。

            下面的例子中,生產者首先必須檢查緩沖是否已滿(numUsedBytes==BufferSize),如果是,線程停下來等待bufferNotFull條件。如果不是,在緩沖中生產數據,增加numUsedBytes,激活條件 bufferNotEmpty。使用mutex來保護對numUsedBytes的訪問。另外,QWaitCondition::wait()接收一個mutex作為參數,這個mutex應該被調用線程初始化為鎖定狀態。在線程進入休眠狀態之前,mutex會被解鎖。而當線程被喚醒時,mutex會處于鎖定狀態,而且,從鎖定狀態到等待狀態的轉換是原子操作,這阻止了競爭條件的產生。當程序開始運行時,只有生產者可以工作。消費者被阻塞等待bufferNotEmpty條件,一旦生產者在緩沖中放入一個字節,bufferNotEmpty條件被激發,消費者線程于是被喚醒。

             const int DataSize = 100000;
             const int BufferSize = 8192;
             char buffer[BufferSize];

             QWaitCondition bufferNotEmpty;
             QWaitCondition bufferNotFull;
             QMutex mutex;
             int numUsedBytes = 0;

             class Producer : public QThread
             {
             public:
                 void run();
             };

             void Producer::run()
             {
                 qsrand(QTime(0,0,0).secsTo(QTime::currentTime()));

                 for (int i = 0; i < DataSize; ++i) {
                     mutex.lock();
                     if (numUsedBytes == BufferSize)
                         bufferNotFull.wait(&mutex);
                     mutex.unlock();

                     buffer[i % BufferSize] = "ACGT"[(int)qrand() % 4];

                     mutex.lock();
                     ++numUsedBytes;
                     bufferNotEmpty.wakeAll();
                     mutex.unlock();
                 }
             }

             class Consumer : public QThread
             {
             public:
                 void run();
             };

             void Consumer::run()
             {
                 for (int i = 0; i < DataSize; ++i) {
                     mutex.lock();
                     if (numUsedBytes == 0)
                         bufferNotEmpty.wait(&mutex);
                     mutex.unlock();

                     fprintf(stderr, "%c", buffer[i % BufferSize]);

                     mutex.lock();
                     --numUsedBytes;
                     bufferNotFull.wakeAll();
                     mutex.unlock();
                 }
                 fprintf(stderr, "\n");
             }

             int main(int argc, char *argv[])
             {
                 QCoreApplication app(argc, argv);
                 Producer producer;
                 Consumer consumer;
                 producer.start();
                 consumer.start();
                 producer.wait();
                 consumer.wait();
                 return 0;
             }

            posted on 2007-08-22 16:48 清源游民 閱讀(13614) 評論(1)  編輯 收藏 引用 所屬分類: Qt

            FeedBack:
            # re: Qt 中的多線程(一)
            2010-12-05 20:29 | kangtian0
            比較了一下,我覺得這個比你的對。
            QWaitCondition

            http://hi.baidu.com/armlinuxhouse/blog/item/d73b59da2b22382533fa1c04.html  回復  更多評論
              
            <2007年9月>
            2627282930311
            2345678
            9101112131415
            16171819202122
            23242526272829
            30123456

            留言簿(35)

            隨筆分類(78)

            隨筆檔案(74)

            文章檔案(5)

            搜索

            •  

            最新評論

            閱讀排行榜

            評論排行榜

            精品国产99久久久久久麻豆| 久久精品中文字幕有码| 无码人妻久久久一区二区三区| 亚洲七七久久精品中文国产| 久久精品国产亚洲αv忘忧草 | 久久AAAA片一区二区| 久久强奷乱码老熟女网站| 国产精品久久久久免费a∨| 激情伊人五月天久久综合| 国产激情久久久久影院老熟女免费| 久久精品无码一区二区三区日韩| 久久久www免费人成精品| 精品久久久久国产免费 | 久久婷婷五月综合97色| 久久国产精品二国产精品| 久久99精品国产自在现线小黄鸭 | 无码国内精品久久人妻麻豆按摩| 久久丫精品国产亚洲av| 日韩欧美亚洲综合久久影院Ds| 久久午夜伦鲁片免费无码| 亚洲精品国产综合久久一线| 久久综合九色综合久99| 中文字幕久久久久人妻| 久久伊人五月天论坛| 国内精品久久久久久不卡影院| 精品久久久久中文字幕日本| 久久精品免费全国观看国产| 久久精品亚洲男人的天堂| 伊人色综合久久| 99久久99久久精品国产片果冻| 国产亚洲欧美精品久久久| 久久天堂AV综合合色蜜桃网| 思思久久精品在热线热| 一本久久a久久精品综合香蕉| 狠狠精品干练久久久无码中文字幕 | 久久乐国产精品亚洲综合| 久久99精品久久久久久野外| 精品欧美一区二区三区久久久 | 7777精品伊人久久久大香线蕉| 无码人妻久久一区二区三区蜜桃| 精品久久人人妻人人做精品 |