Подтвердить что ты не робот

Использование переменной условия в ситуации производителя-потребителя

Я пытаюсь узнать о переменных условий и о том, как использовать его в ситуации производителя-потребителя. У меня есть очередь, где один поток подталкивает числа в очередь, а другой поток выдает числа из очереди. Я хочу использовать переменную условия, чтобы сигнализировать потребляющий поток, когда есть некоторые данные, помещенные производящей нитью. Проблема в том, что есть времена (или чаще всего), которые только подталкивают до двух элементов в очередь, затем зависает. Я указал в функции production(), где он останавливается при работе в режиме отладки. Может ли кто-нибудь помочь мне указать, почему это происходит?

У меня есть следующие глобальные переменные:


boost::mutex mutexQ;               // mutex protecting the queue
boost::mutex mutexCond;            // mutex for the condition variable
boost::condition_variable condQ;

Ниже мой потребительский поток:


void consume()
{
    while( !bStop )   // globally declared, stops when ESC key is pressed
    {
        boost::unique_lock lock( mutexCond );
        while( !bDataReady )
        {
            condQ.wait( lock );
        }

        // Process data
        if( !messageQ.empty() )
        {
            boost::mutex::scoped_lock lock( mutexQ );

            string s = messageQ.front();   
            messageQ.pop();
        }
    }
}

Ниже мой поток производителя:


void produce()
{
    int i = 0;

    while(( !bStop ) && ( i &lt MESSAGE ))    // MESSAGE currently set to 10
    {
        stringstream out;
        out &lt&lt i;
        string s = out.str();

        boost::mutex::scoped_lock lock( mutexQ );
        messageQ.push( s );

        i++;
        {
            boost::lock_guard lock( mutexCond );  // HANGS here
            bDataReady = true;
        }
        condQ.notify_one();
    }
}
4b9b3361

Ответ 1

Вы должны использовать один и тот же мьютекс для защиты очереди, как вы используете в переменной условия.

Это должно быть все, что вам нужно:

void consume()
{
    while( !bStop )
    {
        boost::scoped_lock lock( mutexQ);
        // Process data
        while( messageQ.empty() ) // while - to guard agains spurious wakeups
        {
            condQ.wait( lock );

        }
        string s = messageQ.front();            
        messageQ.pop();
    }
}

void produce()
{
    int i = 0;

    while(( !bStop ) && ( i < MESSAGE ))
    {
        stringstream out;
        out << i;
        string s = out.str();

        boost::mutex::scoped_lock lock( mutexQ );
        messageQ.push( s );
        i++;
        condQ.notify_one();
    }
}