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

了解 Qt 多線程編程 新手必學

移動開發
Qt 作為一種基于 C++ 的跨平臺 GUI 系統,能夠提供給用戶構造圖形用戶界面的強大功能。為了滿足用戶構造復雜圖形界面系統的需求,Qt 提供了豐富的多線程編程支持。

本文介紹的是了解 Qt 多線程編程,本文并沒有過多的去總結多線程的概念,通過了解線程類和一些線程的實現,去真正的弄清楚。首先,我們先來 看文章的講解。

概述

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

Qt 線程類

Qt 包含下面一些線程相關的類:

QThread 提供了開始一個新線程的方法

QThreadStorage 提供逐線程數據存儲

QMutex 提供相互排斥的鎖,或互斥量

QMutexLocker 是一個便利類,它可以自動對QMutex加鎖與解鎖

QReadWriterLock 提供了一個可以同時讀操作的鎖

QReadLocker與QWriteLocker 是便利類,它自動對QReadWriteLock加鎖與解鎖

QSemaphore 提供了一個整型信號量,是互斥量的泛化

QWaitCondition 提供了一種方法,使得線程可以在被另外線程喚醒之前一直休眠。

Qt 高級線程類

QtConcurrent 開啟線程事務

QFutureWatcher 觀測線程狀態

QFuture 線程啟動類

QThread創建線程

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

  1. class MyThread : public QThread  
  2.  {  
  3.      Q_OBJECT  
  4.  protected:  
  5.      void run();  
  6.  };  
  7.  void MyThread::run()  
  8.  {  
  9.      ...  
  10.  } 

之后調用start,Qt即可創建一個線程,并在線程中執行run()函數中代碼,注意UI非線程安全的。

QtConcurrent創建線程

QtConcurrent 創建線程的方法比較多, 而且QtConcurrent 本身比較特殊,若系統有空閑線程時,它會調度空閑線程,無空閑線程時將會創建一個線程。(注意:QtConcurrent 創建線程歸QthreadPool管理,若超過***線程數,將會進入隊列等待),QtConcurrent創建線程的方法多種,以下舉例map函數:

  1. QImage scale(const QImage &image)  
  2.  {  
  3.      qDebug() < < "Scaling image in thread" << QThread::currentThread();  
  4.      return image.scaled(QSize(100, 100), Qt::IgnoreAspectRatio, Qt::SmoothTransformation);  
  5.  }  
  6.  
  7.  int main(int argc, char *argv[])  
  8.  {  
  9.      QApplication app(argc, argv);  
  10.  
  11.      const int imageCount = 20;  
  12.  
  13.      // Create a list containing imageCount images.  
  14.      QList images;  
  15.      for (int i = 0; i < imageCount; ++i)  
  16.          images.append(QImage(1600, 1200, QImage::Format_ARGB32_Premultiplied));  
  17.  
  18.      // Use QtConcurrentBlocking::mapped to apply the scale function to all the  
  19.      // images in the list.  
  20.      QList thumbnails = QtConcurrent::blockingMapped(images, scale);  
  21.  
  22.      return 0;  
  23.  } 

#p#

Qt 線程同步

QMutex, QReadWriteLock, QSemaphore, QWaitCondition 提供了線程同步的手段。使用線程的主要想法是希望它們可以盡可能并發執行,而一些關鍵點上線程之間需要停止或等待。例如,假如兩個線程試圖同時訪問同一個 全局變量,結果可能不如所愿。

QMutex

QMutex 提供相互排斥的鎖,或互斥量。在一個時刻至多一個線程擁有mutex,假如一個線程試圖訪問已經被鎖定的mutex,那么它將休眠,直到擁有mutex的線程對此mutex解鎖。Mutexes常用來保護共享數據訪問。

QReadWriterLock

QReadWriterLock 與QMutex相似,除了它對 “read”,”write”訪問進行區別對待。它使得多個讀者可以共時訪問數據。使用QReadWriteLock而不是QMutex,可以使得多線程程序更具有并發性。

  1. QReadWriteLock lock;  
  2.  void ReaderThread::run()  
  3.  {  
  4.      lock.lockForRead();  
  5.      read_file();  
  6.      lock.unlock();  
  7.  }  
  8.  void WriterThread::run()  
  9.  {  
  10.      lock.lockForWrite();  
  11.      write_file();  
  12.      lock.unlock();  
  13.  } 

QSemaphore

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

  1. const int DataSize = 100000;  
  2.  const int BufferSize = 8192;  
  3.  char buffer[BufferSize];  
  4.  QSemaphore freeBytes(BufferSize);  
  5.  QSemaphore usedBytes;  
  6.  class Producer : public QThread  
  7.  {  
  8.  public:  
  9.      void run();  
  10.  };  
  11.  
  12.  void Producer::run()  
  13.  {  
  14.      qsrand(QTime(0,0,0).secsTo(QTime::currentTime()));  
  15.      for (int i = 0; i < DataSize; ++i) {  
  16.          freeBytes.acquire();  
  17.          buffer[i % BufferSize] = "ACGT"[(int)qrand() % 4];  
  18.          usedBytes.release();  
  19.      }  
  20.  }  
  21.  class Consumer : public QThread  
  22.  {  
  23.  public:  
  24.      void run();  
  25.  };  
  26.  void Consumer::run()  
  27.  {  
  28.      for (int i = 0; i < DataSize; ++i) {  
  29.          usedBytes.acquire();  
  30.          fprintf(stderr, "%c", buffer[i % BufferSize]);  
  31.          freeBytes.release();  
  32.      }  
  33.      fprintf(stderr, "\n");  
  34.  }  
  35.  int main(int argc, char *argv[])  
  36.  {  
  37.      QCoreApplication app(argc, argv);  
  38.      Producer producer;  
  39.      Consumer consumer;  
  40.      producer.start();  
  41.      consumer.start();  
  42.      producer.wait();  
  43.      consumer.wait();  
  44.      return 0;  
  45.  }  

#p#

QWaitCondition

QWaitCondition 允許線程在某些情況發生時喚醒另外的線程。一個或多個線程可以阻塞等待一QWaitCondition ,用wakeOne()或wakeAll()設置一個條件。wakeOne()隨機喚醒一個,wakeAll()喚醒所有。

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

  1. const int DataSize = 100000;  
  2.  const int BufferSize = 8192;  
  3.  char buffer[BufferSize];  
  4.  QWaitCondition bufferNotEmpty;  
  5.  QWaitCondition bufferNotFull;  
  6.  QMutex mutex;  
  7.  int numUsedBytes = 0;  
  8.  
  9.  class Producer : public QThread  
  10.  {  
  11.  public:  
  12.      void run();  
  13.  };  
  14.  void Producer::run()  
  15.  {  
  16.      qsrand(QTime(0,0,0).secsTo(QTime::currentTime()));  
  17.  
  18.      for (int i = 0; i < DataSize; ++i) {  
  19.          mutex.lock();  
  20.          if (numUsedBytes == BufferSize)  
  21.              bufferNotFull.wait(&mutex);  
  22.          mutex.unlock();  
  23.  
  24.          buffer[i % BufferSize] = "ACGT"[(int)qrand() % 4];  
  25.  
  26.          mutex.lock();  
  27.          ++numUsedBytes;  
  28.          bufferNotEmpty.wakeAll();  
  29.          mutex.unlock();  
  30.      }  
  31.  }  
  32.  
  33.  class Consumer : public QThread  
  34.  {  
  35.  public:  
  36.      void run();  
  37.  };  
  38.  void Consumer::run()  
  39.  {  
  40.      for (int i = 0; i < DataSize; ++i) {  
  41.          mutex.lock();  
  42.          if (numUsedBytes == 0)  
  43.              bufferNotEmpty.wait(&mutex);  
  44.          mutex.unlock();  
  45.  
  46.          fprintf(stderr, "%c", buffer[i % BufferSize]);  
  47.  
  48.          mutex.lock();  
  49.          --numUsedBytes;  
  50.          bufferNotFull.wakeAll();  
  51.          mutex.unlock();  
  52.      }  
  53.      fprintf(stderr, "\n");  
  54.  }  
  55.  int main(int argc, char *argv[])  
  56.  {  
  57.      QCoreApplication app(argc, argv);  
  58.      Producer producer;  
  59.      Consumer consumer;  
  60.      producer.start();  
  61.      consumer.start();  
  62.      producer.wait();  
  63.      consumer.wait();  
  64.      return 0;  
  65.  } 

小結:對于Qt 中多線程編程的介紹就到這了,希望本篇文章對你有所幫助。

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

2011-06-16 11:04:07

Qt

2011-06-16 17:19:33

Qt Meego

2011-06-16 11:28:48

Qt QApplicati

2011-06-13 10:03:19

Qt 多線程 編程

2011-06-16 11:13:13

QtQWidget

2011-06-20 13:43:08

Qt Socket 線程

2011-06-22 16:18:23

QT 多線程 QSocket

2013-07-16 10:12:14

iOS多線程多線程概念多線程入門

2023-06-13 13:39:00

多線程異步編程

2009-03-12 10:52:43

Java線程多線程

2011-06-22 10:12:08

Qt 線程

2023-04-02 17:53:10

多線程編程自測

2023-06-07 13:49:00

多線程編程C#

2023-06-05 07:56:10

線程分配處理器

2023-06-06 08:17:52

多線程編程Thread類

2011-06-22 16:08:40

Qt 多線程 事件循環

2011-06-30 17:31:32

Qt 多線程 信號

2011-06-23 14:03:23

2011-06-23 13:20:46

2011-06-16 09:53:25

Qt QML 教程
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 国产一区二区三区四区三区四 | 成人黄色三级毛片 | 美日韩免费视频 | 欧美日韩中文在线 | 日日想夜夜操 | 久操福利| 久久精品无码一区二区三区 | 国产精品一区二区不卡 | 国产精品小视频在线观看 | 中文字幕男人的天堂 | 久久亚洲一区二区 | 久久一区视频 | 国产jizz女人多喷水99 | 国产91视频一区二区 | 日韩一区二区福利视频 | 亚洲欧美综合 | 91精品国产色综合久久 | 亚洲国产一区二区视频 | 98成人网 | 精品欧美一区二区三区久久久 | 欧美 日韩精品 | 天天爽综合网 | 福利视频一区二区 | 北条麻妃一区二区三区在线观看 | 欧美一级特黄aaa大片在线观看 | 亚洲欧美在线一区 | 久久久久久久久久久丰满 | 欧美日韩国产高清 | 国产成人午夜精品影院游乐网 | 九九热精品视频在线观看 | 一区二区视频 | 一本久久a久久精品亚洲 | 欧美a区| 少妇一级淫片免费播放 | 久久久人成影片一区二区三区 | 狠狠操电影 | 91久久久久久久久久久久久 | 91麻豆精品国产91久久久更新资源速度超快 | 美女爽到呻吟久久久久 | 成人午夜影院 | 99久久久无码国产精品 |