Linux C 条件变量阻塞线程用法:等待时CPU占用率为0

一、核心结论

在 Linux 下,条件变量(pthread_cond)+ 互斥锁 是实现「线程阻塞等待变量变化、完全不占用 CPU」的唯一标准方案。线程休眠时不参与调度,CPU 占用为 0;变量变化时可被立即唤醒。


二、核心原理

  1. 阻塞等待

pthread_cond_wait 会自动释放锁并让线程进入休眠阻塞态,CPU 占用率 0%。

2.唤醒机制

修改变量后调用 pthread_cond_signal,内核立即唤醒等待线程。

3.安全保证

互斥锁保护共享变量,避免多线程竞争与脏读。

4.伪唤醒防护

必须用 while 循环判断条件,而非 if


三、最简可运行 C 代码

C 复制代码
#include <stdio.h>
#include <pthread.h>
#include <unistd.h>

// 三要素:锁 + 条件变量 + 等待变量
pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;
pthread_cond_t  cond  = PTHREAD_COND_INITIALIZER;
int             flag  = 0;

// 等待线程:阻塞,0 CPU
void *wait_task(void *arg)
{
    while (1)
    {
        pthread_mutex_lock(&mutex);

        // 阻塞等待 flag 变为 1
        while (flag == 0)
        {
            pthread_cond_wait(&cond, &mutex);
        }

        // 被唤醒
        printf("线程被唤醒\n");
        flag = 0;

        pthread_mutex_unlock(&mutex);
    }
    return NULL;
}

// 唤醒接口:修改变量并唤醒
void trigger(void)
{
    pthread_mutex_lock(&mutex);
    flag = 1;
    pthread_cond_signal(&cond);
    pthread_mutex_unlock(&mutex);
}

// 测试主线程
int main(void)
{
    pthread_t tid;
    pthread_create(&tid, NULL, wait_task, NULL);

    while (1)
    {
        sleep(2);
        trigger();
    }
    return 0;
}

四、Linux 编译与运行

1)编译(必须加 -pthread

Bash 复制代码
gcc cond.c -o cond -pthread

2)运行

Bash 复制代码
./cond

3)运行效果

Plain 复制代码
线程被唤醒
线程被唤醒
线程被唤醒
...

4)查看 CPU(关键验证)

新开终端执行:

Bash 复制代码
top -p $(pidof cond)

等待线程 CPU 占用:0%


五、极简使用模板(工程直接复制)

C 复制代码
// 全局定义
pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;
pthread_cond_t  cond  = PTHREAD_COND_INITIALIZER;
int             ready = 0;

// 等待
void wait_ready(void)
{
    pthread_mutex_lock(&mutex);
    while (ready == 0)
        pthread_cond_wait(&cond, &mutex);
    ready = 0;
    pthread_mutex_unlock(&mutex);
}

// 唤醒
void set_ready(void)
{
    pthread_mutex_lock(&mutex);
    ready = 1;
    pthread_cond_signal(&cond);
    pthread_mutex_unlock(&mutex);
}

六、关键规则(必记)

  1. 必须配合互斥锁使用

  2. 必须用 while 判断条件,防止伪唤醒

  3. pthread_cond_wait 内部会自动解锁/重新加锁

  4. 休眠时不占 CPU ,唤醒时实时响应


七、适用场景

  • 等待数据就绪

  • 等待状态变更

  • 等待事件触发

  • 低功耗、高实时性线程设计

相关推荐
AlfredZhao43 分钟前
生产环境里,为什么不建议把普通端口直接暴露到公网?
linux·https·443·80
戴为沐1 天前
Linux内存扩容指南
linux
zylyehuo2 天前
Linux 彻底且安全地删除文件
linux
用户805533698032 天前
主线 U-Boot 上 RK3506:和闭源 rkbin 拔河的三个隐性契约
linux·嵌入式
用户034095297912 天前
linux fcitx 5 雾凇拼音 设置在中文输入法下仍然输入英文标点
linux
乘云数字DATABUFF2 天前
5分钟部署开源APM Databuff:OpenTelemetry全链路追踪入门实战
运维·后端
Web3探索者4 天前
可视化服务器管理和传统命令行区别是什么?新手教程:Linux 运维到底该用图形界面还是 SSH 命令行?
linux·ssh
zylyehuo4 天前
Linux系统中网线与USB网络共享冲突
linux
荣--4 天前
一键部署不是为了省时间 —— 它是把"买来的 PaaS"变成"自己的平台"的拐点
运维·zabbix·工程化·一键部署·平台化·边界设计
江华森4 天前
动手实战学 Docker — 从零到集群编排完全指南
运维