首页
学习
活动
专区
工具
TVP
发布
精选内容/技术社群/优惠产品,尽在小程序
立即前往

linux c 条件变量

在Linux C编程中,条件变量(Condition Variables)是一种同步机制,用于协调多个线程之间的共享状态。条件变量允许线程等待某个特定条件的成立,从而避免忙等待(busy-waiting),提高系统资源的利用率。

基础概念

条件变量通常与互斥锁(Mutex)一起使用,以确保线程安全地访问共享资源。线程在检查条件是否满足之前,必须先获得互斥锁。如果条件不满足,线程会释放互斥锁并进入等待状态,直到其他线程改变条件并通知它。

相关函数

  • pthread_cond_init(): 初始化条件变量。
  • pthread_cond_wait(): 线程等待条件变量,此函数会自动释放互斥锁,并在被唤醒时重新获得互斥锁。
  • pthread_cond_signal(): 唤醒等待条件变量的一个线程。
  • pthread_cond_broadcast(): 唤醒等待条件变量的所有线程。
  • pthread_cond_destroy(): 销毁条件变量。

优势

  1. 避免忙等待:线程在等待条件成立时不会占用CPU资源。
  2. 线程间通信:允许线程之间以一种松耦合的方式进行通信。
  3. 灵活性:可以精确控制哪些线程被唤醒。

应用场景

  • 生产者-消费者问题:生产者线程生产数据,消费者线程消费数据,条件变量用于协调生产者和消费者之间的速度差异。
  • 读者-写者问题:多个读者线程可以同时读取共享资源,但写者线程必须独占访问,条件变量用于同步读者和写者。

示例代码

以下是一个简单的生产者-消费者模型的示例代码,使用条件变量和互斥锁:

代码语言:txt
复制
#include <pthread.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>

#define BUFFER_SIZE 5

int buffer[BUFFER_SIZE];
int count = 0;

pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;
pthread_cond_t cond_full = PTHREAD_COND_INITIALIZER;
pthread_cond_t cond_empty = PTHREAD_COND_INITIALIZER;

void* producer(void* arg) {
    while (1) {
        int item = rand() % 100;
        pthread_mutex_lock(&mutex);
        while (count == BUFFER_SIZE) {
            pthread_cond_wait(&cond_full, &mutex);
        }
        buffer[count++] = item;
        printf("Produced: %d
", item);
        pthread_cond_signal(&cond_empty);
        pthread_mutex_unlock(&mutex);
        sleep(1);
    }
}

void* consumer(void* arg) {
    while (1) {
        pthread_mutex_lock(&mutex);
        while (count == 0) {
            pthread_cond_wait(&cond_empty, &mutex);
        }
        int item = buffer[--count];
        printf("Consumed: %d
", item);
        pthread_cond_signal(&cond_full);
        pthread_mutex_unlock(&mutex);
        sleep(1);
    }
}

int main() {
    pthread_t producer_thread, consumer_thread;
    pthread_create(&producer_thread, NULL, producer, NULL);
    pthread_create(&consumer_thread, NULL, consumer, NULL);
    pthread_join(producer_thread, NULL);
    pthread_join(consumer_thread, NULL);
    return 0;
}

常见问题及解决方法

  1. 死锁:确保在调用pthread_cond_wait()之前已经获得互斥锁,并且在回调函数中正确处理锁的释放和重新获取。
  2. 虚假唤醒:即使条件没有改变,线程也可能被唤醒。因此,在pthread_cond_wait()之后应该再次检查条件是否满足。
  3. 竞态条件:使用互斥锁保护共享资源,确保在任何时候只有一个线程能够修改共享状态。

通过合理使用条件变量和互斥锁,可以有效地解决多线程编程中的同步问题。

页面内容是否对你有帮助?
有帮助
没帮助

相关·内容

C++ 条件变量使用详解

【导读】:本文主要讲解条件变量的详细使用方法。...condition_variable介绍 在C++11中,我们可以使用条件变量(condition_variable)实现多个线程间的同步操作;当条件不满足时,相关线程被一直阻塞,直到某种条件出现,这些线程才会被唤醒...其主要成员函数如下: 条件变量是利用线程间共享的全局变量进行同步的一种机制,主要包括两个动作: 一个线程因等待"条件变量的条件成立"而挂起; 另外一个线程使"条件成立",给出信号,从而唤醒被等待的线程...这就是条件变量只能和unique_lock一起使用的原因,否则当前线程一直占有锁,线程被阻塞。...条件变量使用 在这里,我们使用条件变量,解决生产者-消费者问题,该问题主要描述如下: 生产者-消费者问题,也称有限缓冲问题,是一个多进程/线程同步问题的经典案例。

2.9K11

【Linux】死锁 | 条件变量部分理解

,就应该将线程在 条件变量中 休眠,防止 不断申请锁 释放锁 当票数满足条件满足时,在将对应的线程唤醒 接口 pthread_cond_init ——初始化 输入 man pthread_cond_init...---- ---- 若将条件变量其定义为全局变量,则可以使用PTHREAD_COND_INITIALIZER 进行初始化 就不用再主函数中使用 初始化 和销毁 了 若条件变量定义为局部变量 ,就必须在主函数中调用...pthread_cond_init 和 pthread_cond_destroy pthread_cond_destroy ——销毁 当条件变量为局部变量时,与pthread_cond_init...要在那个条件变量中等待 第二个参数为 互斥锁 ---- 为什么互斥锁作为参数?...当被唤醒后,会继续向后运行 打印出活动 ---- 在主函数中唤醒 该条件变量下等待的线程 ---- ---- 当主线程每唤醒一个线程,就会打印出对应的活动 5 1 2 3 4 ,在条件变量下进行排队

29731
  • 嵌入式Linux:线程同步(条件变量)

    在Linux环境下,条件变量(Condition Variables)是一种线程同步机制,允许线程在某个条件未满足时进入等待状态,并在其他线程修改共享资源或条件后通知它们。...被唤醒的线程重新获得互斥锁并继续检查条件。 1 条件变量的初始化和销毁 条件变量使用pthread_cond_t数据类型来表示,和互斥锁类似,条件变量在使用前必须初始化。...注意事项: 只能销毁已经初始化的条件变量。 条件变量销毁时,不能有线程在等待它,否则将导致未定义行为。 2 发送信号和等待条件变量 条件变量的核心功能就是发送信号和等待条件。...pthread_mutex_destroy(&mutex); pthread_cond_destroy(&cond); return 0; } Linux中的条件变量是线程同步的强大工具...条件变量不保存状态:如果没有线程在等待条件变量,信号会丢失。

    11310

    C++ std::condition_variable 条件变量用法

    1.简介 condition_variable(条件变量)是 C++11 中提供的一种多线程同步机制,它允许一个或多个线程等待另一个线程发出通知,以便能够有效地进行线程同步。...当线程需要等待某个条件变成真时,它会获取一个互斥锁,然后在条件变量上等待,等待期间会自动释放互斥锁。...如果有多个线程在条件变量上等待,只有其中一个线程会被唤醒,具体是哪个线程 C++ 标准并未明确,所以是不确定的。 被唤醒的线程将尝试获取与条件变量关联的互斥锁,一旦成功获取锁,它可以继续执行。...如果有多个线程在条件变量上等待,所有这些线程都会被唤醒。 唤醒的线程将竞争获取与条件变量关联的互斥锁,然后可以继续执行。...为了解决虚假唤醒和唤醒丢失的问题,需要使用一个变量(通常是 bool 类型的变量)来表示等待的条件,线程在等待前和等待后检查该条件是否满足。

    3.5K21

    【Linux】线程ID与互斥、同步(锁、条件变量)

    Linux线程同步 同步概念与竞态条件 同步:在保证数据安全的前提下,让线程能够按照某种特定的顺序访问临界资源,从而有效避免饥饿问题,叫做同步 竞态条件:因为时序问题,而导致程序异常,我们称之为竞态条件...这种情况就需要用到条件变量。 举例:有A,B,C三个人,一个盘子。B拿出苹果放到盘子上,另外两人就可以到盘子上拿。为了在放苹果的时候,其他人不能来拿,就要加锁,盘子就是临界区。...A,C两人在外面排队,当B放好苹果后就摇铃铛,此时A和C就会根据排队的顺序依次进去拿苹果。 上面的铃铛就是条件变量,人就是线程。摇铃铛后,可以规定是唤醒一个线程还是唤醒全部。...线程条件不满足时,线程就要等待,要在指定的条件变量上等待。 cond:要在这个条件变量上等待 等待完成后,就要进行唤醒。 pthread_cond_signal 表示唤醒一个线程。...pthread_cond_t _c_cond; // 消费者条件变量 // int low_water = _max_cap/3 // int hight_water _max_cap

    11210

    Linux系统编程-(pthread)线程通信(条件变量)

    条件变量介绍 条件变量是线程可用的一种同步机制,条件变量给多个线程提供了一个回合的场所,条件变量和互斥量一起使用,允许线程以无竞争的方式等待特定的条件发生。...条件变量本身是由互斥体保护的,线程在改变条件状态之前必须首先锁住互斥量,其他线程在获取互斥量之前就不会觉察到这种变化,因为互斥量必须锁定之后才改变条件。...条件变量总结: 条件变量要配合互斥锁使用。 条件变量支持单个唤醒和广播方式唤醒。 下面是视频监控的一个项目模型,摄像头的数据使用条件变量保护: 2....条件变量相关接口函数 2.1 条件变量初始化与销毁 #include int pthread_cond_init(pthread_cond_t *restrict cond,const...(&cond, NULL); int pthread_cond_destroy(pthread_cond_t *cond); 成功返回0,否则返回错误码 pthread_cond_init用于初始化条件变量

    2.4K10

    【Linux】线程安全——补充|互斥、锁|同步、条件变量

    : 对变量进行++或者–,在C、C++上看起来只有一条语句,但是汇编之后至少是三条语句: 1.从内存读取数据到CPU寄存器中2.在寄存器中让CPU进行对应的算逻运算3.写回新的结果到内存中变量的位置...在C、C++上,看起来只有一条语句,但是汇编之后至少是三条语句: 1.从内存读取数据到CPU寄存器中 2.在寄存器中让CPU进行对应的算逻运算 3.写回新的结果到内存中变量的位置 对一个资源访问的时候...这种情况就需要用到条件变量 条件变量通常需要配合互斥锁一起使用。 条件变量的使用:一个线程等待条件变量的条件成立而被挂起;另一个线程使条件成立后唤醒等待的线程。...,如果一个人想面试,先得去排队等待区等待,未来所有应聘者都要去条件变量等 条件不满足的时候,线程必须去某些定义好的条件变量上进行等待。...条件变量的使用 通过条件变量来控制线程的执行 条件变量本身不具备互斥的功能,所以条件变量必须配合互斥锁使用: 一次唤醒一个线程 创建2个线程,通过条件变量一秒唤醒一个线程(或者全部唤醒): int tickets

    30020

    条件变量Cond实现

    Cond是什么 下面是wikipedia对条件变量的定义,大体是说条件变量总的来说是等待特定条件的线程的容器。.... ❞ Cond是Go标准库sync包提供的条件变量原语,目的是为等待通知场景下的并发问题提供解决方法。...通俗来说,Cond和某个条件相关,这个条件可以是一个表达式、一个bool变量或是一个函数调用,只要它们的结果是bool类型的值就行。...举个例子,在奥运会100米短跑比赛中,将每个运动员看作一个个goroutine,只有在发令枪响之后,运动员才能开始跑,这里的发令枪响就是条件变量,只有枪响之后,也就是条件满足之后,goroutine才能运行...条件变量Cond实现了这种机制。我们可以使用Cond完成上面的需求。先上改进后的代码,具体API接口作用见下一部分实现原理中的介绍。

    57720

    Linux线程编程同步之互斥锁和条件变量

    通过上面的例子,我们可以看出,条件变量与互斥锁不同,条件变量是用来等待而不是用来上锁的。条件变量用来自动阻塞一个线程,直到某特殊情况发生为止(比如挂号的人都弄完了,它就会自动停止挂号)。...但是通常条件变量和互斥锁同时使用(如上面的例子,各个窗口挂号互不干扰)。条件变量使我们可以睡眠等待某种条件出现。...条件变量是利用线程间共享的全局变量进行同步的一种机制,主要包括两个动作:一个线程等待"条件变量的条件成立"而挂起;另一个线程使"条件成立"(给出条件成立信号)。条件的检测是在互斥锁的保护下进行的。...如果一个条件为假,一个线程自动阻塞,并释放等待状态改变的互斥锁。如果另一个线程改变了条件,它发信号给关联的条件变量,唤醒一个或多个等待它的线程,重新获得互斥锁,重新评价条件。...如果两进程共享可读写的内存,条件变量可以被用来实现这两进程间的线程同步。总之条件变量要和互斥锁一起来用使用。

    1.7K30

    【线程同步】条件变量

    条件变量阻塞等待 条件变量不是锁,它经常和互斥量组合使用。...pthread_cond_t *restrict cond, pthread_mutex_t *restrict mutex, const struct timespec *restrict abstime); /*条件变量阻塞等待...函数参数 cond:条件变量 mutex:互斥锁 abstime:是一个绝对时间,也就是1900年到现在的秒数(在stat函数中介绍过),如果我们要想设置abstime为10秒,应该先获取当前时间,并用这个时间加...函数参数 cond:条件变量 attr:属性 函数返回值 If successful, the pthread_cond_destroy() and pthread_cond_init() functions...唤醒阻塞在条件上的线程 头文件及函数原型 #include /*唤醒阻塞在条件变量cond上的全部线程*/ int pthread_cond_broadcast(pthread_cond_t

    10810

    线程同步-条件变量

    文章目录 引言 条件变量 初始化条件变量:pthread_cond_init 销毁条件变量:pthread_cond_destroy 条件等待:pthread_cond_wait 唤醒等待:pthread_cond_signal...条件变量 一个条件变量是一个pthread_cond_t类型 初始化条件变量:pthread_cond_init int pthread_cond_init(pthread_cond_t *restrict...cond,const pthread_condattr_t *restrict attr); cond:要初始化的条件变量 attr:NULL 销毁条件变量:pthread_cond_destroy...这里引入的铃铛就是条件变量,条件变量必须提供两个东西: 需要一个线程队列 需要有通知机制 此时又来一个C,也是来拿苹果,A和C就会形成竞争了,铃铛想起的时候,就会把A和C都唤醒,这就是pthread_cond_broadcast...确保条件检查的原子性:在多线程环境中,条件变量通常与互斥锁一起使用来保护共享资源。线程在检查条件之前需要持有锁,以避免其他线程修改共享资源。

    9610

    Linux Qt使用POSIX多线程条件变量、互斥锁(量)

    好了,下面以一个实际的例子为背景,来说明Linux POSIX多线程的一些特性。...程序环境:ubuntu 14.04、 Qt 5.5.1、 Posix多线程(C的用法) 这里简单说下我为什么用Linux C的多线程,因为Qt的多线程编程对于一些线程的终止时含糊不清楚的,并且一个线程被终止后的资源是无法被清理的...从条件变量等待中醒来后,会再次获得互斥锁,以操作共享数据。共享数据被操作完成后,再次释放互斥锁。这是我们使用条件变量等待的一个操作流程,如果我们不使用条件变量等待会是怎样的呢?...不使用条件变量等待 ①不使用条件变量等待 如果不使用条件变量等待,则消费者线程在很大一部时间内几乎都是在执行while(1)无限循环,这是很占用CPU资源的,在ubuntu下,使用htop查看的效果如下...②使用条件变量的结果 ? 此时我们看到CPU的占用率是很低的,这也是为什么使用条件变量的原因之一,让不满足的条件的线程挂起,而不是在浪费CPU资源。

    2.2K40

    【C++11】 让多线程开发变得简单--条件变量

    条件变量是C++11中提供的又一种线程同步机制,它可以阻塞一个或者多个线程,直到收到其它线程发出的超时或者通知才能够唤醒正在等待的线程,条件变量需要和互斥量配合使用,在C++ 11中共提供了两种条件变量...条件变量使用过程如下: 获取互斥量 循环检查条件,如果条件不满足则一直阻塞,反之继续执行 线程执行完后调用notify_one或者notify_all唤醒等待的线程 在实际编码中,可以使用条件变量实现一个同步队列...}; 上面的代码实现了一个同步队列的功能,在队列没有满的情况下可以插入数据,如果队列满则会调用m_notFull进行阻塞等待其它线程发送通知,如果队列为空,则不能取出数据,会调用m_notEmpty条件变量进行阻塞

    79310

    【C++】C++11的新特性 — 线程库 ,原子操作 , 条件变量

    获取其中的数据可以使用load接口,修改数据可以使用exchange接口… 3 条件变量 条件变量经常使用在多线程环境下,它允许线程在某些条件不满足时挂起(等待),直到另一个线程更新了共享数据并通知条件变量...条件变量主要提供以下接口: wait():阻塞当前线程,直到条件变量被唤醒,通常在互斥锁锁定的情况下调用,进入wait之前会进行一个解锁!...wait_for():阻塞当前线程,直到条件变量被唤醒或给定的时间超时。 wait_until():阻塞当前线程,直到条件变量被唤醒或到达某个特定的时间点。...条件变量的作用是在变量不符合条件时进行阻塞,等待变量才进行!...变量满足时才进行执行任务!这样就会让不符合条件的变量阻塞在条件变量或者阻塞在获取锁中!通过这样的调控,可以满足多线程情况下的并发需求!

    28610
    领券