linux线程同步-条件变量-附可运行源码

条件变量是用来等待而不是用来上锁的。条件变量用来自动阻塞一个线程,直到某特殊情况发生为止。通常条件变量和互斥锁同时使用。

条件变量使我们可以睡眠等待某种条件出现。条件变量是利用线程间共享的全局变量进行同步的一种机制,主要包括两个动作:

  • 一个线程等待"条件变量的条件成立"而挂起;
  • 另一个线程使"条件成立"(给出条件成立信号)。

条件的检测是在互斥锁的保护下进行的。如果一个条件为假,一个线程自动阻塞,并释放等待状态改变的互斥锁。如果另一个线程改变了条件,它发信号给关联的条件变量,唤醒一个或多个等待它的线程,重新获得互斥锁,重新评价条件。如果两进程共享可读写的内存,条件变量可以被用来实现这两进程间的线程同步。

使用条件变量之前要先进行初始化。可以在单个语句中生成和初始化一个条件变量如:pthread_cond_t my_condition=PTHREAD_COND_INITIALIZER;(用于进程间线程的通信)。可以利用函数pthread_cond_init动态初始化。

条件变量分为两部分: 条件和变量. 条件本身是由互斥量保护的. 线程在改变条件状态前先要锁住互斥量. 它利用线程间共享的全局变量进行同步的一种机制。

一个典型的示例就是生产-消费者模型,生产者生产数据,消费者消费数据,当无数据时,消费者只能等待,当生产者生产了数据,会通知消费者已经有数据,可以消费了。示例代码如下:

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

pthread_cond_t has_data = PTHREAD_COND_INITIALIZER; //初始化条件变量
pthread_mutex_t lock = PTHREAD_MUTEX_INITIALIZER;  //初始化互斥锁

//建立一个公共访问区域
struct msg{
    int num;
    struct msg* next;
};

struct msg *head;

void *producer(void *p)
{
    struct msg* mp;
    while(1)
    {
        mp = (struct msg*)malloc(sizeof(struct msg));
        pthread_mutex_lock(&lock);
        mp->num = rand() % 100 + 1;
        mp->next = head;
        head = mp;
        printf("----Procuder-----tid = %lu,  num=%d\n", pthread_self(),  mp->num);
        pthread_mutex_unlock(&lock);

        pthread_cond_signal(&has_data);


        sleep(rand() % 3);
    }
}

void *consumer(void *p)
{
    struct msg* mp;

    while(1)
    {
        pthread_mutex_lock(&lock);
        while(head == NULL)
        {
            pthread_cond_wait(&has_data, &lock);
        }
        mp = head;
        head = mp->next;
        printf("---Consumer--- tid = %lu, num = %d\n", pthread_self(), mp->num);
        pthread_mutex_unlock(&lock);

        free(mp);
        sleep(rand() % 2);
    }

}

int main()
{
    pthread_t pth_producer;
    pthread_t pth_consumer;
    srand(time(NULL));


    pthread_create(&pth_producer, NULL, producer, NULL);
    pthread_create(&pth_consumer, NULL, consumer, NULL);


    pthread_join(pth_producer, NULL);
    pthread_join(pth_consumer, NULL);

    return 0;
}


结果如下:
在这里插入图片描述

也可关注微信公众号,欢迎交流。
在这里插入图片描述

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值