Linux学习笔记9——pthread_mutex

  • 先来一个不用lock的例子

#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <pthread.h> // pthread头文件
#include <unistd.h>  // sleep的头文件

using namespace std;
int g_total = 0;

// 两个线程
pthread_t thread1;
pthread_t thread2;

// 加10次,每次加1,总共加10
void *add_1(void *arg)
{
    int i = 0;// 故意使用局部变量临时保存g_total以增大出错的可能性
    for(int j = 0; j < 10; j++)
    {
	i = g_total;
	i++;
	sleep(0.5);
	g_total = i;
	cout << __func__ <<"	"<< g_total << endl;
    }
    return NULL;
}

// 加5次,每次加2,总共加10
void *add_2(void *arg)
{
    int i = 0;
    for(int j = 0; j < 5; j++)
    {
	i = g_total;
	i += 2;
	sleep(0.5);
	g_total = i;
	cout << __func__ <<"	"<< g_total << endl;
    }
    return NULL;
}

int main()
{

    // 创建两个个线程。
    pthread_create(&thread1, NULL, &add_1, NULL);// 给全局变量+10
    pthread_create(&thread2, NULL, &add_2, NULL);// 全局变量再+10

    
    // 阻塞主进程,等待两个线程
    pthread_join(thread1, NULL);
    pthread_join(thread2, NULL);


    cout << __func__<<"	"<< g_total << endl;

    return 0;
}

理论上最终g_total的结果应该是20,因为两个线程每个都给他加了10。不过最终的输出结果其实是不确定的,每次执行结果都不一样。其中一次的执行结果如下:

原因就在于没有对全局变量做保护。

  1. 主进程创建出两个线程,在这之后两个线程同时运行;
  2. 假设thread1先执行进循环体,thread2比她稍慢;
  3. thread1先获取全局变量,并将其赋值给i,稍后thread2也做了同样的事,所以当前两个线程里的局部变量i都拿到了全局变量的初始值;
  4. 然后,thread1开始对i做加1运算,稍后thread2也做了同样的事;此时两个线程里的局部变量i的值不相等,一个为1,一个为2;
  5. 两个线程都将i赋值给全局变量,thread1先来,所以g_total变成1,thread2慢一点,g_total变成又再一次被thread2里的i赋值,变成了2;
  6. 所以此时,虽然经过了两次相加的操作,但是g_total并没有先加1再加2,而是加2的操作覆盖了加1的操作,这一点从打印出来的结果上也能看出来,g_total先在thread1里变1,然后在thread2里变2,而不是变3(本来1+2应该变3的)
  • 接下来试试加lock的情况:

#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <pthread.h> // pthread头文件
#include <unistd.h>  // sleep的头文件

using namespace std;
int g_total = 0;
pthread_mutex_t mymutex = PTHREAD_MUTEX_INITIALIZER;

// 两个线程
pthread_t thread1;
pthread_t thread2;

void *add_1(void *arg)
{
    int i = 0;
    for(int j = 0; j < 10; j++)
    {
	pthread_mutex_lock(&mymutex);
	i = g_total;
	i++;
        sleep(0.5);
	g_total = i;
	pthread_mutex_unlock(&mymutex);
	sleep(0.5);
	cout << __func__ <<"	"<< g_total << endl;
    }
    return NULL;
}

void *add_2(void *arg)
{
    int i = 0;
    for(int j = 0; j < 5; j++)
    {
	pthread_mutex_lock(&mymutex);
	i = g_total;
	i += 2;
        sleep(0.5);
	g_total = i;
	pthread_mutex_unlock(&mymutex);
	sleep(0.5);
	cout << __func__ <<"	"<< g_total << endl;
    }
    return NULL;
}

int main()
{

    // 创建两个个线程。
    pthread_create(&thread1, NULL, &add_1, NULL);
    pthread_create(&thread2, NULL, &add_2, NULL);

    
    // 阻塞主进程,等待两个线程
    pthread_join(thread1, NULL);
    pthread_join(thread2, NULL);


    cout << __func__<<"	"<< g_total << endl;

    return 0;
}

注意:

要使用mutex_lock,必须先定义一个pthread_mutex_t类型的全局变量作为标志使用。两个线程都是通过判断这个全局变量的状态来决定是否等待的。

pthread_mutex_t mymutex = PTHREAD_MUTEX_INITIALIZER;

我们给获取全局变量以及给全局变量赋值那一段代码加了锁。之后,在同一个时刻总是只有一个线程可以运行到这块加锁的代码。当thread1运行到循环体内时,thread2就只能等着;同样,当thread2运行到循环体时,thread1也只能等着。

所以,当thread1给全局变量赋值时,thread2什么都没做;当thread2再获取全局变量时,其获取到的就是已经被thread1修改后的变量,而不会使修改前的版本。

例子里我们故意写两个不同的函数,也就是为了说明这种为两个线程加锁的操作,不一定是非要对同一段代码进行,而可以两个线程对应两段不同的代码。线程看到的只是那个mutex_t变量,这是它的唯一判断标准。

加锁之后的执行接过如下:

这回可以看到虽然两个线程交替进行,但是g_total的值一直增加,不会减小了。并且最终的结果也是我们预测的20.

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
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时,会通过条件变量唤醒等待的线程。等待的线程在条件满足时执行任务,并输出计数器的值。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值