linux线程操作

初始化条件变量

int pthread_cond_init(pthread_cond_t *cv,pthread_cond_attr *cattr);
函数返回值:返回0表示成功,返回其他表示失败。
参数:        pthread_cond_attr是用来设置pthread_cond_t的属性,当传入的值是NULL的时候表示使用默认的属性。

函数返回时,创建的条件变量保存在cv所指向的内存中,可以用宏PTHREAD_COND_INITIALIZER来初始化条件变量。值得注意的是不能使用多个线程初始化同一个条件变量,当一个线程要使用条件变量的时候确保它是未被使用的。

 

条件变量的销毁

int pthread_cond_destroy(pthread_cond_t *cv);
返回值:返回0表示成功,返回其他值表示失败。

 

条件变量的使用:

int pthread_cond_wait(pthread_cond_t *cv,pthread_mutex_t *mutex)
int pthread_cond_signal(pthread_cond_t *cv);

 

使用方式如下:

pthread_mutex_lock(&mutex)
while or if(线程执行的条件是否成立)pthread_cond_wait(&cond,&mutex);
线程执行
pthread_mutex_unlock(&mutex);

 

 

为什么要加锁

  1. 线程在执行的部分访问的是进程的资源,有可能多个线程需要访问它,为了避免由于线程并发执行所引起的资源竞争,所以要让每个线程互斥的访问公共资源。
  2. 如果while或if判断不满足线程的执行条件时,线程回调用pthread_cond_wait阻塞自己。pthread_cond_wait被调用线程阻塞的时候,pthread_cond_wait会自动释放互斥锁。线程从调用pthread_cond_wait到操作系统把他放在线程等待队列之后的时候释放互斥锁。

 

使用while和if判断线程执行条件释放成立的区别。

在多线程资源竞争的时候,在一个使用资源的线程里面(消费者)判断资源是否可用,不可用便调用pthread_cond_wait,在另一个线程里面(生产者)如果判断资源可用的话,则会调用pthead_cond_signal发送一个资源可用的信号。

但是在wait成功之后,资源就不一定可以被使用,因为同时有两个或两个以上的线程正在等待次资源,wait返回后,资源可能已经被使用了,在这种情况下

while(resource == FALSE)pthread_cond_wait(&cond,&mutex);

如果之后只有一个消费者,就可使用if。

分解pthread_cond_wait动作为以下步骤:

  1. 线程放在等待队列上,解锁
  2. 等待pthread_cond_signal或者pthread_cond_broadcast信号之后去竞争锁
  3. 若竞争到互斥锁则加锁

 

有可能多个线程在等待这个资源可用的信号,信号发出去之后只有一个资源可用,但是有A,B两个线程在等待,B速度比较快,获得互斥锁,然后加锁,消耗资源,然后解锁,之后A获得互斥锁,但它回去发现资源已经被使用了,它便有两个选择,一个失去访问不存在的资源,另一个就是继续等待,那么等待下去的条件就是使用while,要不然使用if的话pthread_cond_wait返回后,就会顺序执行下去。

 

等待线程:

pthread_cond_wait      前要加锁

pthread_cond_wait      内部会解锁,然后等待条件变量被其他线程激活

pthread_cond_wait      被激活后会再自动加锁

 

激活线程

加锁(和等待线程用同一个锁)

pthread_cond_signal   发送信号(阶跃信号前最后判断有无等待线程)

解锁

激活线程的上面三个操作再运行时间上都是再等待线程的pthread_cond_wait函数内部。

/***
pthread_if.c
***/
#include<stdio.h>
#include<sys/types.h>
#include<stdlib.h>
#include<unistd.h>
#include<pthread.h>pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;
pthread_cond_t cond = PTHREAD_COND_INITIALIZER;int count = 0;void *decrement(void *arg)
{printf("in derment\n");pthread_mutex_lock(&mutex);if(count == 0)pthread_cond_wait(&cond,&mutex);count--;printf("----decrement:%d\n",count);printf("out decrement\n");pthread_mutex_unlock(&mutex);return NULL;
}void *increment(void *arg)
{printf("in increment\n");pthread_mutex_lock(&mutex);count++;printf("-----increment:%d\n",count);if(count != 0){pthread_cond_signal(&cond);}printf("out increment\n");pthread_mutex_unlock(&mutex);return NULL;
}int main()
{pthread_t tid_in,tid_de;pthread_create(&tid_de,NULL,(void*)decrement,NULL);sleep(1);pthread_create(&tid_in,NULL,(void*)increment,NULL);sleep(1);pthread_join(tid_de,NULL);pthread_join(tid_in,NULL);pthread_mutex_destroy(&mutex);pthread_cond_destroy(&cond);return 0;
}

 

 

/***
pthread_while.c
***/
#include<stdio.h>
#include<stdlib.h>
#include<pthread.h>
#include<unistd.h>typedef struct node_s
{int data;struct node_s *next;
}node_t;node_t *head = NULL;pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;
pthread_cond_t cond = PTHREAD_COND_INITIALIZER;void cleanup_handler(void *arg)
{printf("cleanup_handler is running.\n");free(arg);pthread_mutex_unlock(&mutex);
}void *thread_func(void *arg)
{node_t *p = NULL;pthread_cleanup_push(cleanup_handler,p);while(1){pthread_mutex_lock(&mutex);while(NULL == head)pthread_cond_wait(&cond,&mutex);p = head;head = head->next;printf("process %d node\n",p->data);free(p);pthread_mutex_unlock(&mutex);}pthread_cleanup_pop(0);return NULL;
}int main()
{pthread_t tid;node_t *temp = NULL;int i;pthread_create(&tid,NULL,(void*)thread_func,NULL);for(i = 0; i < 10; i++){temp = (node_t*)malloc(sizeof(node_t));temp->data = i;pthread_mutex_lock(&mutex);temp->next = head;head = temp;pthread_cond_signal(&cond);pthread_mutex_unlock(&mutex);sleep(1);}pthread_cancel(tid);pthread_join(tid,NULL);return 0;}

 

转载于:https://www.cnblogs.com/wanghao-boke/p/11613064.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/384804.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Linux下多线程模拟停车场停车

#include<stdio.h> #include<string.h> #include<unistd.h> #include<stdlib.h> #include<pthread.h>#define ONE_SECOND 1000000 #define RANGE 10 #define PERIOD 2 #define NUM_THREADS 4typedef struct {int *carpark; //用一个数组来模…

【C++学习之路】第一章——C++核心方法总论

1 C核心方法总论 1.1 核心思想 通过实际项目来学习编程&#xff0c;更高效掌握编程规则&#xff0c;以及明白各种语法规则的实际应用。 实验思想&#xff1a;任何C的参考资料都不可能覆盖你遇到的所有问题&#xff0c;这个时候&#xff0c;最好的办法就是&#xff0c;编辑代…

【Verilog HDL学习之路】第一章 Verilog HDL 数字设计总论

1 Verilog HDL 数字设计总论 1.1 几个重要的概念 EDA&#xff08;Electronic Design Automation&#xff09; 电子技术自动化 EDA工具 类似于软件工程中的IDE&#xff08;集成开发环境&#xff09;&#xff0c;能够使用Verilog HDL语言描述电路设计&#xff0c;并且能够通过逻…

【学会如何学习系列】从婴儿到大学——学习的本质从未改变过

从婴儿到大学——学习的本质从未改变过 从我们出生一直到现在&#xff0c;其实&#xff0c;学习的本质从来都没有改变过&#xff0c;并且&#xff0c;婴儿时期的我们&#xff0c;是学习能力最强的时候&#xff0c;随着我们不断长大&#xff0c;外界的诱惑越来越多&#xff0c;…

【汇编语言学习之路】第一章 汇编语言核心方法论

版权声明&#xff1a;本学习笔记是本人根据小甲鱼“汇编语言学习课程”和《汇编语言》&#xff08;王爽&#xff09;的书籍&#xff0c;来记录笔记的 1 汇编语言核心方法论 1.1 学习汇编语言的必要性 汇编语言与机器语言是一一对应关系&#xff0c;它的本质是机器语言的代号。…

蓝桥单片机赛题及模拟题代码

链接&#xff1a;https://pan.baidu.com/s/1BVB6VILEed0ufqRDMhvALg 提取码&#xff1a;ukx7

【Verilog HDL学习之路】第二章 Verilog HDL的设计方法学——层次建模

2 Verilog HDL的设计方法学——层次建模 重要的思想&#xff1a; 在语文教学中&#xff0c;应该先掌握核心方法论&#xff0c;再用正确的方法论去做题目&#xff0c;这样能够逐渐加深对于方法论的理解&#xff0c;做题的速度和准确率也会越来越高。在Verilog HDL中&#xff0c…

stm32机械臂资料含视频

这是在网上买的机械臂的资料 含视频及相关软件 在这里分享给大家 不过很大 但是内容很全 链接&#xff1a;https://pan.baidu.com/s/1Fd18ww8jxLH8ChqomstZtw 提取码&#xff1a;147g

【Verilog HDL】第四章 模块的端口连接规则——污水处理模型

先放上连接规则的简图&#xff0c;再详细解释 1. 构建模型——污水处理之流水模型 我们先将上述结构构件一个简单模型&#xff0c;以帮助我们理解。 污水&#xff1a;输入数据净水&#xff1a;输出数据双向数据暂不讨论&#xff0c;取输入和输出的交集即可污水处理厂&…

蓝桥杯嵌入式第七届模拟题 代码

链接&#xff1a;https://pan.baidu.com/s/1fdGC20A51axxPGpoyRL8-w 提取码&#xff1a;by4u

三级嵌入式选择知识点整理

SoC芯片 通用SoC是系统级芯片 既可以是单核 也可以是多核 该芯片中可以包含数字电路 模拟电路 数字模拟混合电路 及射频电路 片上系统可使用单个芯片进行数据的采集 转换 储存 处理 及I/O口功能 智能手机 和平板都使用的SOC WAV是未压缩的数字音频 音质与CD相当 音频视频压缩…

【Verilog HDL】命名的规则研究

Verilog命名规范参考资料 1. 什么可以被命名&#xff1f; 模块的名称模块实例的名称各种数据类型的名称 这些名称我们称之为标识符&#xff0c;标识符的命名规则不再强调&#xff0c;与C语言类似&#xff0c;字母、数字、下划线&#xff08;_&#xff09;和美元符号&#xf…

【Verilog HDL】深入理解部分语法规则的本质

1. 门级描述 统一规则&#xff1a; 门类型 (输出&#xff0c;输入); 细化规则&#xff1a; 与/或门&#xff1a; 多入一出 门 (输出&#xff0c;输入1&#xff0c;输入2,……);缓冲门/非门&#xff1a;一入多出 门 (输出1&#xff0c;输出2,……输出n&#xff0c;输入); 门…

三级嵌入式填空整理

实时 可预测性是实时系统的重要性能标准 按照响应时间 实时操作系统可分为 1.普通实时操作系统 响应时间一般是秒级 2.强实时操作系统 响应时间为毫秒和微秒级 3.弱实时操作系统 响应时间为数十秒 RTOS 响应中断请求并完成相应中断服务子程序的时间非常快 这个时间具有一致性…

【Verilog HDL】从逻辑电路图到门级建模——人工翻译的方法论

从左到右&#xff0c;从上到下 先搞定缓冲/非门&#xff0c;再写与/或门 1. 实例解读 先以四选一数据选择器进行说明 对于数字逻辑的部分不再说明&#xff0c;直接进行逻辑电路图到Verilog门级建模的人工翻译过程的描述。 1.1 端口和线网分析 确定输入/输出端口 输入端口 …

三级嵌入式 汇编指令汇总

ARM条件码 EQ 相等 NE 不相等 CS/HS 无符号大于等于 CC/LO 无符号小于 HI 无符号大于 LS 无符号小于等于 GE 带符号大于等于 L…

【Verilog HDL】语句的并发执行

1. 实践得到的启发 先从一个简单的现象得出结论&#xff0c;Verilog语句是并发执行的&#xff01; 同时&#xff0c;这也是**$monitor系统任务为全局有效**的一个重要支持因素&#xff0c;如果没有并发&#xff0c;它是完不成这项功能的实现的。 众所周知&#xff0c;高级语…

linux下 最常用基本命令

常用命令 基本命令 pwd 打印绝对路径 ls 路径 列举文件名 ls 列举文件的权限 属于哪个用户 容量大小 修改…

【数字逻辑】第四章 组合逻辑电路:端口设计 端口拓展的方法

1. 端口设计的方法 1.1 数据选择器 以四选一数据选择器为例&#xff0c;需要的不同接口类型为 输入端口 数据输入端口地址输入端口使能端&#xff08;控制与拓展&#xff09; 输出端口 数据输出端口 2. 端口拓展的方法——层次建模思想 2.0 两个拓展方向 2.0.1 “数组型…

GCC及Makefile基本使用教程

GCC .c c原始程序 .C/.cc/.cxx c原始程序 .m objective-C原始程序 .i 已经预处理过的c原始程序 .ii 已经预处理过的c原始程序 .s/.S 汇编原始程序 .h 预处理头文件 .o 目标文件 .a/.so 编译后的库文件 -E 生成预处理文件 -S 生成编译过的汇编文件 -c 目标文件 .o -o…