1. // QueueImplementation.cpp : Defines the entry point for the console application.
  2. //
  3. #include "stdafx.h"
  4. #include <windows.h>
  5. #include <iostream>
  6. #include <queue>
  7. #include <string>
  8. #include <process.h>
  9. using namespace std;
  10. struct DataBlock
  11. {
  12. string m_szText;    //sample data
  13. };
  14. class CDataQueue
  15. {
  16. private:
  17. queue<DataBlock>  m_oQueue;   //contains the actual data
  18. CRITICAL_SECTION        m_csData;   //to synchroize access to m_csData among multiple threads
  19. HANDLE                  m_hEvent;   //for signalling presence of absence of data
  20. public:
  21. //create a manual reset event initially signalled.
  22. //This event will be signalled and shall remain so whenever there is data in the queue and
  23. //it shall be reset as long as queue is empty
  24. CDataQueue()
  25. {
  26. InitializeCriticalSection(&m_csData);
  27. m_hEvent = CreateEvent(NULL,TRUE,FALSE,NULL);
  28. };
  29. //close the event handle
  30. ~CDataQueue()
  31. {
  32. DeleteCriticalSection(&m_csData);
  33. CloseHandle(m_hEvent);
  34. };
  35. //public methods to Push data to queue
  36. void Push(DataBlock& oNewData)
  37. {
  38. EnterCriticalSection(&m_csData);
  39. //push new element to queue
  40. m_oQueue.push(oNewData);
  41. //now that there is atleast one element, set the event
  42. SetEvent(m_hEvent);
  43. LeaveCriticalSection(&m_csData);
  44. };
  45. //public methods to Pop data from queue
  46. DataBlock Pop()
  47. {
  48. EnterCriticalSection(&m_csData);
  49. //first get the topmost data block
  50. DataBlock popData = m_oQueue.front();
  51. //next remove it from queue
  52. m_oQueue.pop();
  53. //now, check for new size.. if no more elements in queue
  54. //reset the event
  55. if(!m_oQueue.size())
  56. ResetEvent(m_hEvent);
  57. LeaveCriticalSection(&m_csData);
  58. return popData;
  59. };
  60. //helper method to get the event handle
  61. HANDLE GetEvent(){return m_hEvent;};
  62. };
  63. CDataQueue g_oQueue;
  64. HANDLE     g_hExitEvent;
  65. unsigned __stdcall ProcessData (void * )
  66. {
  67. HANDLE hEvents[] = { g_hExitEvent, g_oQueue.GetEvent()};
  68. DWORD dwRet;
  69. BOOL bContinue = TRUE;
  70. while(bContinue)
  71. {
  72. dwRet = WaitForMultipleObjects(sizeof(hEvents)/sizeof(hEvents[0]),hEvents,FALSE,INFINITE);
  73. switch(dwRet)
  74. {
  75. case WAIT_OBJECT_0 :
  76. {
  77. //exit signalled.. time to quit the thread
  78. bContinue = FALSE;
  79. }
  80. break;
  81. case WAIT_OBJECT_0 + 1:
  82. {
  83. //some data got..
  84. DataBlock oData = g_oQueue.Pop();
  85. //echo data to screen
  86. cout << "Data typed in is .. " << oData.m_szText << "/n";
  87. }
  88. break;
  89. default:break;
  90. }
  91. }
  92. return 0;
  93. }
  94. int main(int argc, char* argv[])
  95. {
  96. DWORD dwThreadID = 0;
  97. HANDLE hThread = NULL;
  98. //create an event to signal worker thread to exit
  99. g_hExitEvent = CreateEvent(NULL,TRUE,FALSE,NULL);//not signalled initially..
  100. //spawn a thread for processing the input
  101. hThread = (HANDLE)_beginthreadex(NULL,0,ProcessData,NULL,0,(unsigned int *)&dwThreadID);
  102. if(hThread)
  103. {
  104. cout << "enter sentence to process /nOR /nenter /"exit/" to quit/n";
  105. do
  106. {
  107. DataBlock oData;
  108. cin >> oData.m_szText;
  109. //if exit typed in.. quit
  110. if(0 == oData.m_szText.compare("exit"))
  111. break;
  112. g_oQueue.Push(oData);
  113. }
  114. while(1);
  115. //time to close ..set the exit event
  116. SetEvent(g_hExitEvent);
  117. //wait for worker thread to close
  118. WaitForSingleObject(hThread,INFINITE);
  119. //close the thread handle
  120. CloseHandle(hThread);
  121. }
  122. CloseHandle(g_hExitEvent);
  123. return 0;
  124. }
  1. template<typename Data>
  2. class concurrent_queue
  3. {
  4. private:
  5. std::queue<Data> the_queue;
  6. mutable boost::mutex the_mutex;
  7. boost::condition_variable the_condition_variable;
  8. public:
  9. void push(Data const& data)
  10. {
  11. boost::mutex::scoped_lock lock(the_mutex);
  12. the_queue.push(data);
  13. lock.unlock();
  14. the_condition_variable.notify_one();
  15. }
  16. bool empty() const
  17. {
  18. boost::mutex::scoped_lock lock(the_mutex);
  19. return the_queue.empty();
  20. }
  21. bool try_pop(Data& popped_value)
  22. {
  23. boost::mutex::scoped_lock lock(the_mutex);
  24. if(the_queue.empty())
  25. {
  26. return false;
  27. }
  28. popped_value=the_queue.front();
  29. the_queue.pop();
  30. return true;
  31. }
  32. void wait_and_pop(Data& popped_value)
  33. {
  34. boost::mutex::scoped_lock lock(the_mutex);
  35. while(the_queue.empty())
  36. {
  37. the_condition_variable.wait(lock);
  38. }
  39. popped_value=the_queue.front();
  40. the_queue.pop();
  41. }
  42. };

boost 线程安全队列的更多相关文章

  1. 使用Condition Variables 实现一个线程安全队列

    使用Condition Variables实现一个线程安全队列 测试机: i7-4800MQ .7GHz, logical core, physical core, 8G memory, 256GB ...

  2. Boost线程库学习笔记

    一.创建一个线程 创建线程 boost::thread myThread(threadFun); 需要注意的是:参数可以是函数对象或者函数指针.并且这个函数无参数,并返回void类型. 当一个thre ...

  3. Boost线程详解

    一.创建一个线程 创建线程 boost::thread myThread(threadFun); 需要注意的是:参数可以是函数对象或者函数指针.并且这个函数无参数,并返回void类型. 当一个thre ...

  4. Boost 线程学习笔记

    Bolg转载自:http://www.cnblogs.com/lvdongjie/p/4447193.html 一: 创建线程 #include <iostream> #include & ...

  5. Linux多线程系列-2-条件变量的使用(线程安全队列的实现)

    多线程情况下,往往需要使用互斥变量来实现线程间的同步,实现资源正确共享. linux下使用如下变量和函数 //条件变量 pthread_cond_t int pthread_cond_init (pt ...

  6. BOOST 线程完全攻略 - 基础篇

    http://blog.csdn.net/iamnieo/article/details/2908621 2008-09-10 12:48 9202人阅读 评论(3) 收藏 举报 thread多线程l ...

  7. BOOST 线程完全攻略 - 扩展 - 可被关闭的线程类

    本文假设读者已经基本了解boost线程库的使用方法. boost是个开源工程,线程这一块也在不断完善之中,到现在这个阶段,boost::thread仅仅实现了一个完美的技术框架,但是读者在实际使用中会 ...

  8. 线程池 队列 synchronized

    线程池 BlockingQueue synchronized volatile 本章从线程池到阻塞队列BlockingQueue.从BlockingQueue到synchronized 和 volat ...

  9. Java线程安全队列BlockingQueue

    线程安全队列BlockingQueue 用法跟普通队列没有区别,只是加入了多线程支持. 这里主要说说add和put,以及poll和take的区别: add和put都是用来忘队列里面塞东西的,而poll ...

随机推荐

  1. HTTP 1.1学习笔记

    前言 由于HTTP 1自身的局限性,它不能很好的为用户提供性能良好的WEB服务.于1999年6月正式发布了HTTP1.1标准REC2616,它厘清了之前版本中很多有歧义的地方,而且还新增了很多重要的优 ...

  2. 造轮子和用轮子:快速入门JavaScript模块化

    造轮子和用轮子:快速入门JavaScript模块化 前言 都说“不重复造轮子”,就像iPhone——它除了打电话还可以播放音乐——但是工程师不用从零开始做一个音乐播放功能,也许只要在iPhone的系统 ...

  3. 有关Math数学运算的js函数

    随机函数;  Moth.random()   //3.以下通过循环给数组每个元素赋值,随机数. // Math.random(); 可以随机0~1之间的任意数 [0,1) // alert(Math. ...

  4. 使用jdk中提供的排序方式

    package com.bjpowernode.t01; import java.util.Arrays; /** * 使用jdk中提供的排序方式 * */public class TestArray ...

  5. Mysql查询出所有列名

    select group_concat(COLUMN_NAME Separator ',') as COLUMN_NAME from information_schema.COLUMNS where ...

  6. (第8篇)实时可靠的开源分布式实时计算系统——Storm

    摘要: 在Hadoop生态圈中,针对大数据进行批量计算时,通常需要一个或者多个MapReduce作业来完成,但这种批量计算方式是满足不了对实时性要求高的场景.那Storm是怎么做到的呢? 博主福利 给 ...

  7. Codeforces 585D Lizard Era: Beginning

    Lizard Era: Beginning 折半之后搜就完事了, 直接存string字符串卡空间, 随便卡卡空间吧. #include<bits/stdc++.h> #define LL ...

  8. 一种使用pyinstaller时图标问题解决方案

    一种使用pyinstaller时图标问题解决方案 0x00 场景   使用pyinstaller将.py文件编译成.exe文件时,想要使用自己心仪的图标(.ico)比较麻烦.在使用pyinstalle ...

  9. 学机器学习,不会数据处理怎么行?—— 二、Pandas详解

    在上篇文章学机器学习,不会数据处理怎么行?—— 一.NumPy详解中,介绍了NumPy的一些基本内容,以及使用方法,在这篇文章中,将接着介绍另一模块——Pandas.(本文所用代码在这里) Panda ...

  10. java8 Optional正确使用姿势

    Java 8 如何正确使用 Optional import lombok.Data; import lombok.EqualsAndHashCode; import lombok.ToString; ...