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);
为什么要加锁
- 线程在执行的部分访问的是进程的资源,有可能多个线程需要访问它,为了避免由于线程并发执行所引起的资源竞争,所以要让每个线程互斥的访问公共资源。
- 如果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动作为以下步骤:
- 线程放在等待队列上,解锁
- 等待pthread_cond_signal或者pthread_cond_broadcast信号之后去竞争锁
- 若竞争到互斥锁则加锁
有可能多个线程在等待这个资源可用的信号,信号发出去之后只有一个资源可用,但是有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 = ; void *decrement(void *arg)
{
printf("in derment\n");
pthread_mutex_lock(&mutex);
if(count == )
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 != )
{
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();
pthread_create(&tid_in,NULL,(void*)increment,NULL);
sleep(); pthread_join(tid_de,NULL);
pthread_join(tid_in,NULL);
pthread_mutex_destroy(&mutex);
pthread_cond_destroy(&cond);
return ;
}
/***
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()
{
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();
return NULL;
} int main()
{
pthread_t tid;
node_t *temp = NULL;
int i;
pthread_create(&tid,NULL,(void*)thread_func,NULL); for(i = ; i < ; 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();
}
pthread_cancel(tid);
pthread_join(tid,NULL);
return ; }
linux线程操作的更多相关文章
- linux 线程操作问题undefined reference to 'pthread_create'的解决办法(cmake)
问题原因: pthread 库不是 Linux 系统默认的库,连接时需要使用静态库 libpthread.a. 所以在使用pthread_create()创建线程时,需要链接该库. 1. 终端:问题解 ...
- 8)Linux程序设计入门--线程操作
)Linux程序设计入门--线程操作 前言:Linux下线程的创建 介绍在Linux下线程的创建和基本的使用. Linux下的线程是一个非常复杂的问题,由 于我对线程的学习不时很好,我在这里只是简单的 ...
- linux线程
线程:轻量级进程,在资源.数据方面不需要进行复制 不间断地跟踪指令执行的路径被称为执行路线 进程的结构:task_struck:地址空间 线程:轻量级的进程 在同一个进程中创建的线程,在共享进程的地址 ...
- [转载]Linux 线程实现机制分析
本文转自http://www.ibm.com/developerworks/cn/linux/kernel/l-thread/ 支持原创.尊重原创,分享知识! 自从多线程编程的概念出现在 Linux ...
- linux线程的实现
首先从OS设计原理上阐明三种线程:内核线程.轻量级进程.用户线程 内核线程 内核线程就是内核的分身,一个分身可以处理一件特定事情.这在处理异步事件如异步IO时特别有用.内核线程的使用是廉价的,唯一使用 ...
- linux线程的实现【转】
转自:http://www.cnblogs.com/zhaoyl/p/3620204.html 首先从OS设计原理上阐明三种线程:内核线程.轻量级进程.用户线程 内核线程 内核线程就是内核的分身,一个 ...
- Linux线程-创建
Linux的线程实现是在内核以外来实现的,内核本身并不提供线程创建.但是内核为提供线程[也就是轻量级进程]提供了两个系统调用__clone()和fork (),这两个系统调用都为准备一些参数,最终都用 ...
- Linux线程学习(一)
一.Linux进程与线程概述 进程与线程 为什么对于大多数合作性任务,多线程比多个独立的进程更优越呢?这是因为,线程共享相同的内存空间.不同的线程可以存取内存中的同一个变量.所以,程序中的所有线程都可 ...
- Linux线程学习(二)
线程基础 进程 系统中程序执行和资源分配的基本单位 每个进程有自己的数据段.代码段和堆栈段 在进行切换时需要有比较复杂的上下文切换 线程 减少处理机的空转时间,支持多处理器以及减少上下文切换开销, ...
随机推荐
- MyEclipse的 lib和Build path(构建路径)
首先两种方式对于放置jar包的方式是不同的: Build path(构建路径):对于种方式来说,可以算是对jar包文件的一个引用.可以引用lib下的jar包,也可以引用本地磁盘上的jar包. WEB- ...
- vmware下的linux没有网络问题解决思路
这里几天一直在搞vm下的linux主机的网络问题,这里做个总结 这里使用的NAT连接方式 1.首先保证本机联网正常 2.检查虚拟机相应的服务(VMware NET Service 和 VMware W ...
- 广州CBC2019
CBC2019-day1 25 August 2019 on 学术前沿huyujia 8月24日上午,CBC2019正式开幕.主持人首先对大会情况以及与会嘉宾做了简要介绍:紧接着,校领导.大会主席以及 ...
- tcp协议close_wait与time_wait状态含义
题目描述 1.什么是三次握手,四次挥手?为什么分别要三次与四次? 2.tcp协议中,close_wait与time_wait状态分别代表什么含义,为什么要设计这两种状态,解决了什么问题? 3.time ...
- webpack---eslint-loader学习随笔
第一步.在项目内安装: npm i -D eslint eslint-loader . 第二步.webpack的配置loader 注意loader的顺序,需先执行eslint-loader. 第三步. ...
- PHP 中使用ajax时一些常见错误总结整理
这篇文章主要介绍了PHP 中使用ajax时一些常见错误总结整理的相关资料,需要的朋友可以参考下 PHP作为后端时,前端js使用ajax技术进行相互信息传送时,经常会出错误,对于新手来说有些手足无措.总 ...
- js sort() 排序用法(转载)
原文:https://blog.csdn.net/m0_37885651/article/details/80016718 sort() 方法用于对数组的元素进行排序,并返回数组.默认排序顺序是根据字 ...
- Python——元组
是为了满足,某些值当被定义以后就不可修改或删除而出现的元组形式. 特点: 元组中的元素不可被修改或删除 没有独立的功能 可以进行嵌套,当嵌套方可以修改删除时,可以对嵌套方进行. 元组可以进行公共功能中 ...
- 16.centos7基础学习与积累-002
1.从头开始积累centos7系统运用 大牛博客:https://blog.51cto.com/yangrong/p5 互联网公司服务器品牌: dell 服务器品牌: 1U=4.45CM 2010年以 ...
- linux系统编程之信号(五)
今天继续对信号进行学习,开始正入正题: sigaction函数: 安装信号之前我们已经学过一个函数:signal,它最早是在unix上出现的,它是对不可靠信号进行安装的,之后出现了可靠信号和实时信号, ...