线程加锁与同步,pthread_mutex_lock,pthread_mutex_unlock,pthread_cond_wait和pthread_cond_signal

        线程之间的同步是通过临界资源来实现的。简言之,临界资源就是全局变量,而同步就是一个线程使另个线程的条件成立,唤醒阻塞的线程,然后将CPU控制权给它,此时为了防止线程混乱,最好将自己的条件变为假,等待另一个线程使自己的条件为真。

注意事项:

       1)一个线程调用pthread_cond_wait()阻塞等待时,后面的代码不会再执行。而当另一线程调用唤醒该线程后,该线程继续从阻塞的地方向下执行。

        2)一个线程调用pthread_cond_signal()唤醒另外的线程使,自己后面的代码将继续执行。

        3)在线程调用pthread_cond_wait()阻塞的时候,该线程必须处于加锁状态,即调用pthread_mutex_lock()。

        4)在线程调用pthread_cond_signal()唤醒其他线程,把CPU控制权给出去的时候,自身一定要调用pthread_mutex_unlock()进行解锁。

        其他不多分享,运行代码看结果自然明白。

代码详解:

#include<pthread.h>
#include<unistd.h>
#include<stdio.h>
#include<stdlib.h>

pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;/*初始化互斥锁*/
pthread_cond_t  cond = PTHREAD_COND_INITIALIZER;//init cond

void *thread1(void*);
void *thread2(void*);

int i = 1; //global

int main(void){
    pthread_t t_a;
    pthread_t t_b;//two thread

    pthread_create(&t_a,NULL,thread2,(void*)NULL);
    pthread_create(&t_b,NULL,thread1,(void*)NULL);//Create thread
    
    printf("t_a:0x%x, t_b:0x%x:\n", t_a, t_b);
    pthread_join(t_b,NULL);//wait a_b thread end
    pthread_mutex_destroy(&mutex);
    pthread_cond_destroy(&cond);
    exit(0);
}

void *thread1(void *junk)
{   
	printf("call thread1 \n");
    
    while(i<9)
	{
        if(i%2 == 0)
		{
			pthread_mutex_lock(&mutex); //互斥锁
			printf("thread1:******i=%d\n", i);
			printf("thread1: sleep i=%d\n", i);
			printf("thread1: sleep i=%d******end\n", i);
			i++;
			pthread_cond_signal(&cond); // 发信号唤醒其他线程,剩余代码继续执行,当沉睡的时候唤醒其他线程。其他线程执行完继续执行剩余代码
			printf("thread1 give up to 2\n");
			pthread_mutex_unlock(&mutex);
			sleep(1);
		}
    }
}

void *thread2(void*junk)
 {
    while(i < 9)
    {	
        if(i%2 != 0)
		{
			pthread_mutex_lock(&mutex);
			printf("*** call thread2 \n");
			printf("*** thread2: %d\n",i);
			printf("*** thread2: sleep i=%d\n", i);
			printf("*** thread2: sleep i=%d******end\n", i);
			i++;
			pthread_cond_wait(&cond,&mutex); //wait 马上把权限给出去,不执行剩余代码
			printf("thread2 have been weak up!\n");
			pthread_mutex_unlock(&mutex);
			
			sleep(1);
		}
    }
}

 

  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
pthread_mutex_init: 用于初始化互斥锁(mutex),将互斥锁设置为默认属性。 ```c #include <pthread.h> int pthread_mutex_init(pthread_mutex_t *mutex, const pthread_mutexattr_t *attr); ``` pthread_cond_init: 用于初始化条件变量(condition variable),将条件变量设置为默认属性。 ```c #include <pthread.h> int pthread_cond_init(pthread_cond_t *cond, const pthread_condattr_t *attr); ``` pthread_create: 用于创建一个新的线程,并执行指定的函数。 ```c #include <pthread.h> int pthread_create(pthread_t *thread, const pthread_attr_t *attr, void *(*start_routine)(void*), void *arg); ``` pthread_cond_signal: 用于唤醒一个等待在条件变量上的线程。 ```c #include <pthread.h> int pthread_cond_signal(pthread_cond_t *cond); ``` pthread_mutex_unlock: 用于解锁互斥锁,允许其他线程获取该互斥锁。 ```c #include <pthread.h> int pthread_mutex_unlock(pthread_mutex_t *mutex); ``` pthread_mutex_lock: 用于加锁互斥锁,如果互斥锁已经被锁定,则调用线程会阻塞直到互斥锁被解锁。 ```c #include <pthread.h> int pthread_mutex_lock(pthread_mutex_t *mutex); ``` 下面是一个使用这些函数的简单示例: ```c #include <stdio.h> #include <pthread.h> pthread_mutex_t mutex; pthread_cond_t cond; int count = 0; void* thread_func(void* arg) { pthread_mutex_lock(&mutex); // 检查条件是否满足 while (count < 10) { // 等待条件变量被唤醒 pthread_cond_wait(&cond, &mutex); } // 条件满足,执行任务 printf("Thread: Count is now %d\n", count); pthread_mutex_unlock(&mutex); return NULL; } int main() { pthread_t thread; pthread_mutex_init(&mutex, NULL); pthread_cond_init(&cond, NULL); // 创建线程 pthread_create(&thread, NULL, thread_func, NULL); // 模拟更新计数 for (int i = 0; i < 15; ++i) { pthread_mutex_lock(&mutex); count++; // 每次更新计数后,唤醒等待的线程 if (count >= 10) { pthread_cond_signal(&cond); } pthread_mutex_unlock(&mutex); } // 等待线程结束 pthread_join(thread, NULL); pthread_mutex_destroy(&mutex); pthread_cond_destroy(&cond); return 0; } ``` 在上面的示例中,主线程更新一个计数器,当计数器达到10时,会通过条件变量唤醒等待的线程。等待的线程在条件满足时执行任务,并输出计数器的值。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值