当前位置: 首页 > 知识库问答 >
问题:

当线程不应该阻塞时

程城
2023-03-14

我正在尝试创建一个消费者-生产者程序,其中消费者线程生产者的数字填充数组,消费者线程打印填充数组的数字。目前,我可以填充数组并在使用者/生产者线程之间来回传递数据,但我希望生产者创建数字的速度比使用者处理数字的速度快。

此刻,每1秒产生一个数字,每3消耗一个数字。在消耗一个之前应该产生两个数字,但是我的生产者线程正在等待,直到它产生的数字被消耗。

我试过移动互斥锁和解锁,还有信号,但我没有得到它的工作。下面的代码产生以下输出:

#include <stdio.h>
#include <stdlib.h>
#include <time.h>
#include <pthread.h>
#include <unistd.h>

struct Data {
  int num;
  int wait_time;
}; 


pthread_mutex_t the_mutex;
pthread_cond_t condc, condp;
//int count = 0;
struct Data buffer[32];


void* producer(void *ptr) {
  int i, w;             /* counter and random wait time */
  struct Data data;
  int count = 0;

  while(1) {
    //w = rand() % 5 + 3;
    w = 1;
    sleep(w);               /* Wait between 3 and 7 seconds */

    data.num = rand() % 1000;       /* Create random number to pass */
    //data.wait_time = rand() % 8 + 2;
    data.wait_time = 3;

    pthread_mutex_lock(&the_mutex); /* lock the buffer */

    while (buffer[count].num != 0) {        /* while full */
      //pthread_cond_signal(&condc);
      pthread_cond_wait(&condp, &the_mutex);
    }

    //pthread_mutex_lock(&the_mutex);   /* lock the buffer */
    buffer[count] = data;

    pthread_cond_signal(&condc);    /* signal consumer */
    pthread_mutex_unlock(&the_mutex);

    printf("Produced %i and slept for %i seconds\n", buffer[count].num, w);

    if (count != 31){
      count += 1;
      //printf("Producer count: %i\n", count);
    }
    else
      count = 0;

    //pthread_cond_signal(&condc);  /* signal consumer */
    //pthread_mutex_unlock(&the_mutex); /* unlock */
  }
  pthread_exit(0);
}


void* consumer(void *ptr) {
  int i;
  int count = 0;

  //for(i = 1; i <= MAX; i++) {
  while(1) {   
     pthread_mutex_lock(&the_mutex);    /* lock th buffer */

    while(buffer[count].num == 0){
      //pthread_cond_signal(&condp);        /* while empty */
      pthread_cond_wait(&condc, &the_mutex);
    }

    //pthread_mutex_lock(&the_mutex);

    sleep(buffer[count].wait_time);
    printf("Consumed %i and slept for %i seconds\n", buffer[count].num, buffer[count].wait_time);

    //pthread_mutex_lock(&the_mutex);
    buffer[count].num = 0;
    buffer[count].wait_time = 0;

    pthread_cond_signal(&condp);    /* signal producer */
    pthread_mutex_unlock(&the_mutex);

    if(count != 31){
      count += 1;
      //printf("Consumer count: %i\n", count);
    }
    else
      count = 0;

    //pthread_cond_signal(&condp);  /* signal producer */
    //pthread_mutex_unlock(&the_mutex); /* unlock */
  }
  pthread_exit(0);
}


int main(int argc, char **argv) {
  pthread_t pro, con;
  srand(time(NULL));

  for (int i = 0; i < 32; i++) {        /* Initialize buffer */
    buffer[i].num = 0;
    buffer[i].wait_time = 0;
  }

  // Initialize the mutex and condition variables
  /* What's the NULL for ??? */
  pthread_mutex_init(&the_mutex, NULL); 
  pthread_cond_init(&condc, NULL);      /* Initialize consumer condition variable */
  pthread_cond_init(&condp, NULL);      /* Initialize producer condition variable */

  // Create the threads
  pthread_create(&con, NULL, consumer, NULL);
  pthread_create(&pro, NULL, producer, NULL);

  // Wait for the threads to finish
  // Otherwise main might run to the end
  // and kill the entire process when it exits.
  pthread_join(con, NULL);
  pthread_join(pro, NULL);
  //pthread_join(&con, NULL);
  //pthread_join(&pro, NULL);

  // Cleanup -- would happen automatically at end of program
  pthread_mutex_destroy(&the_mutex);    /* Free up the_mutex */
  pthread_cond_destroy(&condc);     /* Free up consumer condition variable */
  pthread_cond_destroy(&condp);     /* Free up producer condition variable */

}

输出(程序在1秒后打印第1行,然后在3秒后同时打印第2行和第3行):

Produced 985 and slept for 1 seconds
Consumed 985 and slept for 3 seconds
Produced 540 and slept for 1 seconds

我更希望输出看起来像这样:

Produced 985 and slept for 1 seconds
Produced 540 and slept for 1 seconds
Consumed 985 and slept for 3 seconds

共有1个答案

万选
2023-03-14

使用者锁定互斥体,然后休眠3秒。因此,生产者必须等待消费者完成它的工作/睡眠,然后才能生产其他产品。当锁就位时,避免休眠任何一个线程。

编辑:只是编辑了您的代码一点,这似乎可以工作没有信号等在这里。试试看:

#include <stdio.h>
#include <stdlib.h>
#include <time.h>
#include <pthread.h>
#include <unistd.h>

struct Data {
    int num;
    int wait_time;
}; 

pthread_mutex_t the_mutex;
pthread_cond_t condc, condp;
struct Data buffer[32];

void* producer(void *ptr) {
    int i, w;             /* counter and random wait time */
    struct Data data;
    int count = 0;

    while(1) {
        printf("prod count %d\n",count);
        w = 1;
        sleep(w);
        data.num = rand() % 1000;
        data.wait_time = 3;

        while (buffer[count].num != 0) {
            printf("buffer full, count = %d\n", count);
            sleep(1);
        }

        // Only using the mutex when we want to change the variable.
        pthread_mutex_lock(&the_mutex); 
        buffer[count] = data;
        pthread_mutex_unlock(&the_mutex);

        printf("Produced %i and slept for %i seconds\n", buffer[count].num, w);

        if (count != 31){
            count += 1;
        }
        else
            count = 0;
    }
    pthread_exit(0);
}

void* consumer(void *ptr) {
    int i;
    int count = 0;

    while(1) {       /* lock th buffer */
        printf("cons count %d\n",count);
        while(buffer[count].num == 0){
            printf("buffer empty, count = %d\n", count);
            sleep(1);
        }

        sleep(buffer[count].wait_time);
        printf("Consumed %i and slept for %i seconds\n", buffer[count].num, buffer[count].wait_time);

        pthread_mutex_lock(&the_mutex);
        buffer[count].num = 0;
        buffer[count].wait_time = 0;
        pthread_mutex_unlock(&the_mutex);

        if(count != 31){
            count += 1;
            //printf("Consumer count: %i\n", count);
        }
        else {
            count = 0;
        }
    }
    pthread_exit(0);
}


int main(int argc, char **argv) {
    pthread_t pro, con;
    srand(time(NULL));

    for (int i = 0; i < 32; i++) {        /* Initialize buffer */
        buffer[i].num = 0;
        buffer[i].wait_time = 0;
    }

    // Initialize the mutex and condition variables
    /* What's the NULL for ??? */
    pthread_mutex_init(&the_mutex, NULL); 
    pthread_cond_init(&condc, NULL);      /* Initialize consumer condition variable */
    pthread_cond_init(&condp, NULL);      /* Initialize producer condition variable */

    // Create the threads
    pthread_create(&con, NULL, consumer, NULL);
    pthread_create(&pro, NULL, producer, NULL);

    // Wait for the threads to finish
    // Otherwise main might run to the end
    // and kill the entire process when it exits.
    pthread_join(con, NULL);
    pthread_join(pro, NULL);
    //pthread_join(&con, NULL);
    //pthread_join(&pro, NULL);

    // Cleanup -- would happen automatically at end of program
    pthread_mutex_destroy(&the_mutex);    /* Free up the_mutex */
    pthread_cond_destroy(&condc);     /* Free up consumer condition variable */
    pthread_cond_destroy(&condp);     /* Free up producer condition variable */
}
 类似资料:
  • 我有4-5个工作线程处理大型消息队列。我还有另一段代码,它使用2-3个worker运行。我想在处理大型消息队列时阻止所有其他工作者。 我正在使用JDK6和Jms 编辑: 队列进程工作者从未终止。当没有消息时,它们阻塞队列。这些工作者由执行器线程池管理,如果我使用读写锁,其中一个工作者也会被阻塞。此外,如果使用循环屏障,那么我必须终止线程,以便重新传递阻塞的第二个进程。由于工作者是由线程池管理的,所

  • 我编写了一个简单的类,我计划将其扩展为客户端套接字编程应用程序的一部分。类涉及一个BlockingQueue(我从这里复制了代码:相当于Java的BlockingQueue的C++)。当我创建了下面的包装类的一个实例后,我打算让它生成一个单独的线程,该线程只需执行BlockingQueue上阻塞的printer()函数,直到有一个或多个字符串可用,然后它只需将字符串打印到控制台窗口。在我的预期应用

  • 背景我使用log4j2(2.12.1)与同步根和异步记录器。Lmax环形缓冲区大小默认为256*1024。我在控制台的appender。我用JSON布局记录MapMessage。我的日志消息的平均大小约为100字节。 有了上面的细节,我注意到很少有线程被阻塞在 我的问题是。。 环形缓冲区是否很快就满了,这导致主线程背压(在我的例子中,servlet容器线程是http-nio-8080-exec-1

  • 我试图使用neo4j中的命令从CSV文件中导入大约500,000行数据。 下面是我正在使用的代码: 期望的行为:第一次出现以另一个艺术家为特征的人会创建关系,并且应该将关系的属性设置为1。对于随后的每一次事件,强度属性将增加1。因此,经常以艺术家B为特色的艺术家A应该具有类似的关系 关系是方向性的,在这种情况下,方向性很重要(A以B为特征不同于B以A为特征)。 在确定问题时,是否有其他有用的信息?

  • 线程实例的join()方法可用于将一个线程的执行开始“连接”到另一个线程的执行结束,这样一个线程在另一个线程结束之前不会开始运行。如果对线程实例调用join(),则当前运行的线程将阻塞,直到线程实例完成执行 但是如果我有多个线程并且当我在循环内部调用join时。所有线程并行运行。但是根据连接的概念,首先连接的线程应该完成,然后只有主线程才允许连接其他线程。 } 在上面的代码中,如果第一个线程被连接

  • 编辑: 主要问题:为什么只有一个线程抛出interruptedexception,而两个线程都阻塞在条件上。await 所以下面的代码只是我创建的一个示例。主要的问题是开发一个生产者-消费者实现,在这个实现中,我必须创建一个模拟类,它产生两种线程:客户线程和厨师线程,这两种线程是基于可重入锁进行同步的。在执行一些操作(客户添加订单,厨师执行服务这些订单)后,我调用客户线程上的join以确保所有订单