pthread_cond_wait用法解析与案例

pthread_cond_wait用法解析与案例

    pthread_cond_wait(pthread_cond_t *cond, pthread_mutex_t *mutex)函数传入的参数mutex用于保护条件,因为我们在调用pthread_cond_wait时,如果条件不成立我们就进入阻塞,但是进入阻塞这个期间,如果条件变量改变了的话,那我们就漏掉了这个条件。因为这个线程还没有放到等待队列上,所以调用pthread_cond_wait前要先锁互斥量,即调用pthread_mutex_lock(),pthread_cond_wait在把线程放进阻塞队列后,自动对mutex进行解锁,使得其它线程可以获得加锁的权利。这样其它线程才能对临界资源进行 访问并在适当的时候唤醒这个阻塞的进程。当pthread_cond_wait返回的时候又自动给mutex加锁。

      pthread_cond_wait必须放  在pthread_mutex_lockpthread_mutex_unlock之间,因为他要根据共享变量的状态来决定是否要等待,而为了不永远等待下去所以必须要在lock/unlock队中 

以下为案例:

    实现多线程编程,主程序终端输入命令如:1,2,3,4等,线程send_pthread实现

将命令解析为cmd1,cmd2,cmd3,quit等,线程wake_pthread实现将命令写入文件新建文件中(如:log.txt)

要求:要用到线程同步与互斥机制

编译:./a.out pthead.c  -lpthread

调试结果:


 

  vi   log.txt




#include<stdio.h>
#include <pthread.h>
#include<stdlib.h>
#include<string.h>
#include<errno.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>

typedef struct _Linknode_
{
	int data;
	struct _Linknode_ *next;

}Linknode;

Linknode *create_linknode()
{
	Linknode *node = NULL;
	node = (Linknode *)malloc(sizeof(Linknode));
	node->next = NULL;
	return node;
}

int insert_linknode(Linknode *head,int data)
{
	Linknode *new = create_linknode();
	new->data = data;
	new->next = head->next;
	head->next = new;
	return 0;
}

int delete_linknode(Linknode *head)
{
	int value;
	Linknode *p = head->next;
	value = p->data;
	head->next = p->next;
	free(p);
	return value;
}


int  is_empty_linknode(Linknode *head)
{
	return (head->next == NULL) ?1:0;
}


pthread_mutex_t send_lock,wake_lock;
pthread_cond_t  send_cond,wake_cond;

char *content[] = {"cmd1","cmd2","cmd3","quit",NULL};
int INDEX ;

int init_pthread_lock(pthread_mutex_t *mutex_lock)
{
	int ret;

	ret = pthread_mutex_init(mutex_lock,NULL);
	if(ret != 0 )
	{
		fprintf(stderr,"fail to pthread_mutex_init\n");
		exit(EXIT_FAILURE);
	}
	return 0;
}

int init_pthread_cond(pthread_cond_t *cond)
{
	int ret ;

	ret = pthread_cond_init(cond,NULL);
	if(ret != 0)
	{
		fprintf(stderr,"fail to pthread_cond_init\n");
		exit(EXIT_FAILURE);
	}

	return 0;
}

void *send_pthread(void *arg)
{
	int fd = *((int *)arg);
	while(1)	
	{
		pthread_cond_wait(&send_cond,&send_lock);
		write(fd,content[INDEX],strlen(content[INDEX]) );
		pthread_mutex_unlock(&send_lock);

		if(INDEX == 3)
			break;
	}

	pthread_exit(NULL);
}

void *wake_thread(void *arg)
{
	Linknode *L = (Linknode *)arg;
	while(1)
	{
		if(is_empty_linknode)
			pthread_cond_wait(&wake_cond,&wake_lock);
		
		switch(delete_linknode(L))
		{
			case 1:
				INDEX = 0;
				break;
			case 2:
				INDEX = 1;
				break;
			case 3:
				INDEX = 2;
				break;
			case 4:
				INDEX = 3;
				goto next;
		}
		pthread_mutex_unlock(&wake_lock);

		pthread_mutex_lock(&send_lock);
		pthread_cond_signal(&send_cond);
		pthread_mutex_unlock(&send_lock);
		
		usleep(500);
	}
next :
		pthread_cond_signal(&send_cond);
		pthread_exit(NULL);
}

int main(int argc, const char *argv[])
{
	int fd;
	int cmd;
	int ret;
	pthread_t  tid[2];
	Linknode *head = create_linknode();


	if(argc < 2)
	{
		fprintf(stderr,"Usage : %s filename\n",argv[0]);
		exit(EXIT_FAILURE);
	}
	
	if((fd = open(argv[1],O_WRONLY | O_CREAT | O_TRUNC,0666)) < 0)
	{
		fprintf(stderr,"fail to open %s,%s\n", argv[1],strerror(errno));
		exit(EXIT_FAILURE);
	}
    
	ret = pthread_create(&tid[0],NULL,send_pthread,(void *)&fd);
	if(ret != 0)
	{
		fprintf(stderr,"fail to pthread_create send_pthread,%s\n",strerror(errno));
		exit(EXIT_FAILURE);
	}

	ret = pthread_create(&tid[1],NULL,wake_thread,(void *)head);
	if(ret != 0)
	{
		fprintf(stderr,"fail to pthread_create wake_thread ,%s\n",strerror(errno));
		exit(EXIT_FAILURE);
	}

	init_pthread_lock (&send_lock);
	init_pthread_lock(&wake_lock);
	init_pthread_cond(&send_cond);
	init_pthread_cond(&send_cond);


	while(1)
	{
		printf("Input cmd>");
		scanf("%d",&cmd);

		insert_linknode(head,cmd);
		pthread_cond_signal(&wake_cond);
		
		if(cmd  == 4)
			break;
	}
	
	pthread_join(tid[0],NULL);
	pthread_join(tid[1],NULL);

	exit(EXIT_SUCCESS);
}



  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值