Skip to content

Instantly share code, notes, and snippets.

@mikeando
Created July 20, 2010 01:55
Show Gist options
  • Star 5 You must be signed in to star a gist
  • Fork 4 You must be signed in to fork a gist
  • Save mikeando/482342 to your computer and use it in GitHub Desktop.
Save mikeando/482342 to your computer and use it in GitHub Desktop.
#ifndef ma_concurrent_queue_h
#define ma_concurrent_queue_h
// Based on code from http://www.justsoftwaresolutions.co.uk/threading/implementing-a-thread-safe-queue-using-condition-variables.html
// Original version by Anthony Williams
// Modifications by Michael Anderson
#include "boost/thread.hpp"
#include <deque>
template<typename Data>
class concurrent_queue
{
private:
std::deque<Data> the_queue;
mutable boost::mutex the_mutex;
boost::condition_variable the_condition_variable;
bool is_canceled;
//All threads that might be waiting on this queue
boost::thread_group * waiting_threads;
public:
concurrent_queue( boost::thread_group * waiting_thread_group ) : the_queue(), the_mutex(), the_condition_variable(), is_canceled(false), waiting_threads(waiting_thread_group) {}
~concurrent_queue() { cancel(); waiting_threads->join_all(); }
struct Canceled{};
void push(Data const& data)
{
boost::mutex::scoped_lock lock(the_mutex);
if (is_canceled) throw Canceled();
the_queue.push_back(data);
lock.unlock();
the_condition_variable.notify_one();
}
bool empty() const
{
boost::mutex::scoped_lock lock(the_mutex);
if (is_canceled) throw Canceled();
return the_queue.empty();
}
bool try_pop(Data& popped_value)
{
boost::mutex::scoped_lock lock(the_mutex);
if (is_canceled) throw Canceled();
if(the_queue.empty())
{
return false;
}
popped_value=the_queue.front();
the_queue.pop_front();
return true;
}
void wait_and_pop(Data& popped_value)
{
boost::mutex::scoped_lock lock(the_mutex);
while(the_queue.empty() && !is_canceled)
{
the_condition_variable.wait(lock);
}
if (is_canceled) throw Canceled();
popped_value=the_queue.front();
the_queue.pop_front();
}
std::deque<Data> wait_and_take_all()
{
boost::mutex::scoped_lock lock(the_mutex);
while(the_queue.empty() && !is_canceled)
{
the_condition_variable.wait(lock);
}
if (is_canceled) throw Canceled();
std::deque<Data> retval;
std::swap(retval, the_queue);
return retval;
}
void cancel()
{
boost::mutex::scoped_lock lock(the_mutex);
if (is_canceled) throw Canceled();
is_canceled = true;
lock.unlock();
the_condition_variable.notify_all();
}
};
#endif
#include "producer_consumer_queue.hpp"
#include <iostream>
void consumer( concurrent_queue<int> * queue )
{
try
{
while(true)
{
int i;
std::cout<<"About to wait"<<std::endl;
queue->wait_and_pop(i);
std::cout<<"Got me an "<<i<<std::endl;
}
}
catch( const concurrent_queue<int>::Canceled & e )
{
return;
}
}
int main()
{
{
boost::thread_group workers;
concurrent_queue<int> queue( &workers) ;
workers.add_thread( new boost::thread(consumer, &queue) );
queue.push(7);
sleep(1); // This gives the other thread a chance to wait again..
}
}
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment