QT 信号量QSemaphore的使用

生产者——消费者模型中,生产者和消费者线程之间需要传递一定量的数据,两个线程会使用一个特定大小的共享环形缓冲器。
生产者向缓冲器中写入数据,直到它到达缓冲器的终点;然后它会再次从起点重新开始,覆盖已经存在的数据。消费者线程则会读取生成的数据。
在生产者——消费者实例中,对于同步的需求有两个部分:如果生产者线程生成数据的速度太快,那么将会把消费者线程还没有读取的数据覆盖;如果消费者线程读取数据的速度过快,那么它就会越过生产者线程而读取一些垃圾数据。
解决这一问题的一个粗略方法是,让生产者线程填满缓冲器,然后等待消费者线程读取完缓冲器中全部速度。
另一个更有效的方案是使用两个信号量。
freeSpace信号量控制生产者线程写入数据的那部分缓冲器,usedSpace信号量则控制消费者线程读取数据的那部分缓冲器区域。这两个区域是相互补充的。常用缓冲区容量值初始化freeSpace信号量,意味着它最多可以获取的缓冲器资源量。在启动这个应用程序时,消费者线程就会获得自由的字节并把它们转换为用过的字节。用0初始化usedSpace信号量,以确保消费者线程不会在一开始就读取到垃圾数据。
在生产者线程中,每次反复写入都是从获取一个自由字节开始。如果该缓冲器中充满了消费者线程还没有读取的数据,那么对acquire()的调用就会被阻塞,直到消费者线程开始消费这些数据。一旦生产者线程获取这一字节,就写入数据,并将这个字节释放为用过的字节,以让消费者线程读取到。
在消费者线程中,我们从获取一个用过的字节开始。如果缓冲器中还没有任何可用的数据,那么将会阻塞对acquire()调用,直到生产者线程生产数据。一旦获取到这个字节,就使用数据,并把字节释放为自由的字节,这样,生产者线程就可以再次写入。

生产者把自由的空间转换为用过的空间,消费者将用过的空间转换为自由的空间。

[cpp] view plain copy 

  1. /**************************************
  2. *说明:生产者——消费者线程模型
  3. ***************************************/
  4. #include <QtCore/QCoreApplication>
  5. #include <QSemaphore>
  6. #include <QThread>
  7. #include <stdio.h>
  8. //
  9. QSemaphore usedSem(0);//已用量
  10. QSemaphore unusedSem(4096);//未使用变量
  11. unsigned int buffer[4096];//缓冲区
  12. //生产者线程
  13. class producer:public QThread
  14. {
  15. public:
  16. producer();
  17. private:
  18. void run();
  19. };
  20. producer::producer()
  21. {
  22. ;
  23. }
  24. //
  25. void producer::run()
  26. {
  27. int i=0;
  28. for(i=0;i<10000;i++)
  29. {
  30. unusedSem.acquire();//空闲信号量减1
  31. buffer[i%4096]=i;
  32. usedSem.release();//已用信号量加1
  33. }
  34. }
  35. //消费者线程
  36. class customer:public QThread
  37. {
  38. public:
  39. customer();
  40. private:
  41. void run();
  42. };
  43. customer::customer()
  44. {
  45. ;
  46. }
  47. void customer::run()
  48. {
  49. int i=0;
  50. for(i=0;i<10000;i++)
  51. {
  52. usedSem.acquire();//已用信号量减1
  53. printf("%d\n",buffer[i%4096]);
  54. unusedSem.release();//空闲信号量加1
  55. }
  56. }
  57. //
  58. int main(int argc, char *argv[])
  59. {
  60. QCoreApplication a(argc, argv);
  61. //
  62. producer producerThread;
  63. customer customerThread;
  64. //
  65. producerThread.start();
  66. customerThread.start();
  67. //
  68. producerThread.wait();
  69. customerThread.wait();
  70. //
  71. return a.exec();
  72. }
时间: 2024-11-10 01:28:25

QT 信号量QSemaphore的使用的相关文章

基类子类在Qt信号量机制下的思考

背景知识: 基类 superClass class superClass { public: superClass(){ std::cout << "superClass()" << std::endl; } virtual ~superClass(){} virtual void print() { std::cout << "superclass print " << std::endl; } }; 子类subCl

QT开发(三十四)——QT多线程编程

QT开发(三十四)--QT多线程编程 一.QT多线程简介 QT通过三种形式提供了对线程的支持,分别是平台无关的线程类.线程安全的事件投递.跨线程的信号-槽连接. QT中线程类包含如下: QThread 提供了开始一个新线程的方法    QThreadStorage 提供逐线程数据存储    QMutex 提供相互排斥的锁,或互斥量    QMutexLocker 是一个辅助类,自动对 QMutex 加锁与解锁    QReadWriterLock 提供了一个可以同时读操作的锁    QReadL

QT多线程学习

一.想要使用Qthread必须先创建,继承Qthread的类. 1 #ifndef THREADTEST_H 2 #define THREADTEST_H 3 4 #include <QThread> 5 #include <QMutex> 6 7 class ThreadTest : public QThread 8 { 9 public: 10 ThreadTest(); 11 ~ThreadTest(); 12 13 void setMessage(const QString

35.QT-多线程

程序和进程的区别 进程是动态的,程序是静态的 进程是暂时的,程序是永久的, 进程是通过程序运行时得到的 程序是一个数据文件,进程是内存中动态的运行实体,用来存储数据段,代码段,指针等 程序和进程的关系 一个程序可能对应多个进程 一个进程可能包含多个程序,比如一个程序依赖多个其它动态库时 进程和线程的关系 进程是操作系统资源分配的基本单位 线程是操作系统调度执行的基本单位 每个进程包含了1个至多个线程,并且每个线程都可以共享进程的资源 线程也是CPU调度和分派的基本单位,它是比进程更小的能独立运行

Qt同步线程(比较清楚,而且QMutex QMutexLocker QReadWriteLock QSemaphore QWaitCondition 每个都有例子)

Qt同步线程 我们知道,多线程有的时候是很有用的,但是在访问一些公共的资源或者数据时,需要进行同步,否则会使数据遭到破坏或者获取的值不正确.Qt提供了一些类来实现线程的同步,如QMutex,QMutexLocker,QReadWriteLock,QReadLocker,QWriteLocker,QSemaphore和QWaitCondition.下面我们分别来看它们的用法: QMutex 首先,简单的了解一下QMutex提供的函数. 构造函数:QMutex ( RecursionMode mod

Qt线程的简单使用(四)——QSemaphore的使用

参考资料:示例来源    http://www.cnblogs.com/venow/archive/2012/10/15/2724943.html Qt 帮助手册 QSemaphore QSemaphore提供了一定数量的信号量. 一个 semaphore信号量是一种泛化的mutex.一个mutex只能被锁住一次,但是semaphore被获取多次.信号量通常被用于保护一定数量的同种资源. Semaphore提供两种基本的操作,acquire() and release(): Acquire(n)

Qt同步线程(QMutex QMutexLocker QReadWriteLock QSemaphore QWaitCondition )

Qt同步线程 我们知道,多线程有的时候是很有用的,但是在访问一些公共的资源或者数据时,需要进行同步,否则会使数据遭到破坏或者获取的值不正确.Qt提供了一些类来实现线程的同步,如QMutex,QMutexLocker,QReadWriteLock,QReadLocker,QWriteLocker,QSemaphore和QWaitCondition.下面我们分别来看它们的用法: QMutex 首先,简单的了解一下QMutex提供的函数. 构造函数:QMutex ( RecursionMode mod

Qt多线程(有详细例子)

Qt线程类 Qt 包含下面一些线程相关的类:QThread 提供了开始一个新线程的方法QThreadStorage 提供逐线程数据存储QMutex  提供相互排斥的锁,或互斥量QMutexLocker 是一个便利类,它可以自动对QMutex加锁与解锁QReadWriterLock 提供了一个可以同时读操作的锁QReadLocker与QWriteLocker 是便利类,它自动对QReadWriteLock加锁与解锁QSemaphore 提供了一个整型信号量,是互斥量的泛化QWaitConditio

QT 多线程程序设计(也有不少例子)

QT通过三种形式提供了对线程的支持.它们分别是,一.平台无关的线程类,二.线程安全的事件投递,三.跨线程的信号-槽连接.这使得开发轻巧的多线程Qt程序更为容易,并能充分利用多处理器机器的优势.多线程编程也是一个有用的模式,它用于解决执行较长时间的操作而不至于用户界面失去响应.在Qt的早期版本中,在构建库时有不选择线程支持的选项,从4.0开始,线程总是有效的. 线程类 Qt 包含下面一些线程相关的类:QThread 提供了开始一个新线程的方法QThreadStorage 提供逐线程数据存储QMut