Linux下线程的互斥与同步详解

news2024/11/28 4:44:44

🤖个人主页:晚风相伴-CSDN博客

💖如果觉得内容对你有帮助的话,还请给博主一键三连(点赞💜、收藏🧡、关注💚)吧

🙏如果内容有误或者有写的不好的地方的话,还望指出,谢谢!!!

让我们共同进步

下一篇《生产者消费者模型》敬请期待

目录

🔥线程间互斥的相关概念

💪互斥量的接口

初始化互斥量

销毁互斥量

互斥量的加锁与解锁

🔥探究互斥量实现原理

可重入函数和线程安全 

两者的概念区分 

常见的线程不安全和安全情况

可重入与线程安全的联系与区别

☀死锁 

产生死锁的四个必要条件

避免死锁

🔥线程同步 

条件变量 

同步的概念与竞态条件

🔥条件变量接口

初始化 

销毁条件 

条件等待

唤醒等待

🔥解释pthread_cond_wait中的互斥量


🔥线程间互斥的相关概念

  • 临界资源:多线程执行流共享的资源就叫做临界资源
  • 临界区:每个线程内部,访问临界资源的代码,就叫做临界区
  • 互斥:任何时刻,互斥保证有且只有一个执行流进入临界区,访问临界资源,通常对临界资源其保护作用
  • 原子性:不会被任何调度机制打断的操作,该操作只有两种状态,要么完成,要么未完成。

先来看看下面简单实现的抢票的代码

int tickets = 1000;

 void* getTickets(void* args)
 {
     (void)args;
     while(true)
     {
         if(tickets > 0)
         {
             usleep(1000);
             printf("%p: %d\n", pthread_self(), tickets);
             tickets--;
         }
         else
         {
            break;
         }
     }
     return nullptr;
 }

 int main()
 {
     pthread_t t1, t2, t3;
     pthread_create(&t1, nullptr, getTickets, nullptr);
     pthread_create(&t1, nullptr, getTickets, nullptr);
     pthread_create(&t1, nullptr, getTickets, nullptr);

     pthread_join(t1, nullptr);
     pthread_join(t2, nullptr);
     pthread_join(t3, nullptr);

     return 0;
 }

结果演示

💪为什么结果会出现-1呢?

原因:首先要知道一个线程什么时候被调度,调度多长时间,完全是有计算机确定的,程序员决定不了。tickets在进行减减操作时,是分三步的

①读取数据到CPU内的寄存器中

②CPU内部进行计算--

③将结果写回内存中

为了方便叙述,这里给线程编个号

一号线程来了,由于时间片很短执行到第②步就被切走了,二号线程来了,它没有被打断,所以它执行完了这三步,并且这个线程的优先级比较高,一直执行tickets--操作,直到tickets减到1停止,在执行到第①步的时候被切走了,而一号线程回来了,继续从它被打断的地方继续向后执行,也就是从第②步开始继续向后执行,在写回内存后,tickets已经减到了1,但是这个线程又把tickets修改为了999,并且这时它的时间片很长,所以这次又一直将tickets减到了1,由于判断条件tickets不为0,所以tickets继续减减操作,此时tickets减为了0,此时二号线程来了,将0读入到寄存器中进行减减操作,所以结果出现了-1,这就导致了问题的出现。

 要解决上面的问题,就需要做到以下三点:

  1. 代码必须要有互斥行为:当代码进入临界区执行时,不允许其它线程进入临界区。
  2. 如果多个线程同时要求执行临界区的代码,并且临界区没有线程在执行,那么只能允许一个线程进入该临界区。
  3. 如果线程不在临界区中执行,那么该线程不能阻止其它线程进入临界区。

要做到以上三点,就需要一把互斥锁,将临界区资源锁住,没有拿到钥匙的线程就不能访问临界区资源,这就能做到保护了临界区资源。Linux上提供的这把互斥锁叫互斥量。

💪互斥量的接口

初始化互斥量

有两种方式初始化互斥量

方法一:全局初始化分配

pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER

方法二:局部初始化分配

int pthread_mutex_init(pthread_mutex_t *restrict mutex, const pthread_mutexattr_t *restrict attr);

参数:

  • mutex:要初始化的互斥量
  • attr:nullptr

返回值:成功返回0,失败返回错误码

销毁互斥量

 int pthread_mutex_destroy(pthread_mutex_t *mutex);

参数:

  • mutex:要销毁的互斥量

返回值:成功返回0,失败返回错误码

 销毁互斥量时需要注意

  • 使用全局初始化的互斥量不需要销毁
  • 不要销毁一个已经加锁的互斥量
  • 已经销毁的互斥量要确保后面的代码中不再有加锁的操作

互斥量的加锁与解锁

int pthread_mutex_lock(pthread_mutex_t *mutex);
int pthread_mutex_unlock(pthread_mutex_t *mutex);

返回值:成功返回0,失败返回错误码

 调用pthread_mutex_lock加锁时,可能会遇到以下情况:

  • 互斥量还没被加锁,处于未锁定状态,那么调用该函数会将互斥量加锁锁定。
  • 在调用该函数之前,其它线程已经申请了锁,锁定了该互斥量,或者存在其它线程同时竞争式的申请互斥量,但没有竞争到互斥量,那么调用pthread_mutex_lock就会被阻塞,等待会吃两解锁。

所以将上面的抢票代码修改如下:

int tickets = 1000; // 临界资源

class ThreadData
{
public:
    ThreadData(string &name, pthread_mutex_t *pmtx) : _tname(name), _pmtx(pmtx)
    {
    }

public:
    string _tname;
    pthread_mutex_t *_pmtx;
};


void *getTickets(void *args)
{
    ThreadData* td = (ThreadData*)args;
    while (true)
    {
        int n = pthread_mutex_lock(td->_pmtx); // 加锁保护临界区资源
        assert(n == 0);
        if (tickets > 0)
        {
            usleep(1000);
            printf("%s : %d\n", td->_tname.c_str(), tickets);
            cout << td->_tname << " : " << tickets << endl;
            tickets--;
            n = pthread_mutex_unlock(td->_pmtx);
            assert(n == 0);
        }
        else
        {
            n = pthread_mutex_unlock(td->_pmtx);
            assert(n == 0);
            break;
        }
        // 处理后续的动作
        cout << "恭喜,抢票成功" << endl;
        usleep(1000);
    }

    return nullptr;
}

#define THREAD_NUM 5

int main()
{
    pthread_mutex_t mtx;
    pthread_mutex_init(&mtx, nullptr); // 局部定义的锁进行初始化的形式
    pthread_t tid[THREAD_NUM];
    for (int i = 0; i < THREAD_NUM; i++)
    {
        string name = "thread ";
        name += to_string(i + 1);
        ThreadData *td = new ThreadData(name, &mtx);
        pthread_create(tid + i, nullptr, getTickets, (void *)td);
    }

    for (int i = 0; i < THREAD_NUM; i++)
    {
        pthread_join(tid[i], nullptr);
    }

    pthread_mutex_destroy(&mtx); // 最后将锁释放掉
    return 0;
}

 结果演示:

🔥探究互斥量实现原理

加锁的目的是保证操作的原子性。 从汇编的角度来看,如果只有一条汇编语句,我们就认为该汇编语句的执行是原子的, 在汇编中给我们提供了swap或者exchange指令,该指令的作用是将内存中的数据与CPU内寄存器中的数据(CPU内寄存器中的数据也叫做执行流的上下文,寄存器的空间是被所有执行流锁共享的,但是里面的数据是被某一个执行流私有的)进行交换,由于只有一条指令,所以可以保证其原子性。

解锁时会把互斥量变为1。

可重入函数和线程安全 

两者的概念区分 

线程安全:多个线程并发执行同一段代码时,不会出现不同的结果。
重入:同一个函数被不同的执行流调用,当前一个执行流还没有执行完,就有其它的执行流再次进入该函数,我们称这种情况是重入。一个函数在重入的情况下,运行结果不会出现任何不同或者任何问题,则该函数被称为可重入函数,否则称为不可重入函数。

常见的线程不安全和安全情况

不安全情况:

  • 不保护共享变量的函数
  • 函数状态随着被调用,状态发生变化的函数
  • 返回指向静态变量指针的函数
  • 调用线程不安全函数的函数

 安全情况:

  •  每个线程对全局变量或者静态变量只有读取权限,而没有写入权限,一般来说这些线程是安全的。
  • 类或者接口对于线程来说都是原子操作的
  • 多个线程之间的切换不会导致该接口的执行结果存在二义性

可重入与线程安全的联系与区别

联系:

  • 函数是可重入的,那就是线程安全的。
  • 函数是不可重入的,那就不能由多个线程使用,有可能引发线程安全问题。
  • 如果一个函数中有全局变量,那么这个函数既不是线程安全也不是可重入的。

区别:

  • 可重入函数是线程安全函数的一种
  • 线程安全不一定是可重入的,而可重入函数则一定是线程安全的
  • 如果对临界资源的访问加上锁,则这个函数是线程安全的,但如果这个可重入函数的锁还未释放则会产生死锁,因此是不可重入的。

☀死锁 

死锁是指子在一组进程中的各个进程均占有不会释放的资源,但因互相申请被其它进程所占用不会释放的资源而处于的一种永久等待状态。

产生死锁的四个必要条件

  • 互斥条件:一个资源每次只能被一个执行流使用
  • 请求与保持条件:一个执行流因请求支援而阻塞时,对已获得的资源保持不放
  • 不剥夺条件:一个执行流已获得的资源,在未使用完之前,不能被强行剥夺
  • 循环等待条件: 若干执行流之间形成一种头尾相接的循环等待资源的关系

避免死锁

  • 破坏死锁的四个必要条件
  • 加锁顺序一致
  • 避免锁未释放的场景
  • 资源一次性分配
  • 对死锁检测
  • 银行家算法

🔥线程同步 

条件变量 

当我们申请临界资源前,要先检测临界资源是否存在,做检测的本质也是在访问临界资源,所以对临界资源的检测一定是要在加锁和解锁之间的。例如一个线程访问队列时,发现队列为空,那么它只能等待,直到其它线程将一个节点添加到队列中,在检测队列是否为空时,如果该线程一直轮询检测,那么势必要频繁的申请锁和释放锁,这样太浪费资源了,那么这种情况就需要用到条件变量了。

因此条件变量可以让线程不在频繁的自己检测了,当第一次检测到条件不满足时就挂起等待,当条件满足时,再通知该线程,让它来申请资源和访问。

同步的概念与竞态条件

同步:在保证数据安全的前提下,让线程能够按照某种特定的顺序访问临界资源,从而有效的解决了访问临界资源的合理性问题。

竞态条件:因为时序问题,而导致程序异常,我们称之为竞态条件。

🔥条件变量接口

初始化 

和互斥量那里一样分为全局初始化和局部初始化

局部初始化 

int pthread_cond_init(pthread_cond_t *restrict cond,const pthread_condattr_t *restrict attr);

参数

  • cond:要初始化的条件变量
  • attr:设置为nullptr即可

返回值:成功返回0,失败返回错误码

全局初始化

pthread_cond_t cond = PTHREAD_COND_INITIALIZER; 

销毁条件 

int pthread_cond_destroy(pthread_cond_t *cond) ;

条件等待

int pthread_cond_wait(pthread_cond_t *restrict cond,pthread_mutex_t *restrict mutex);
参数

  • cond:要在这个条件变量上等待
  • mutex:互斥量

唤醒等待

int pthread_cond_broadcast(pthread_cond_t *cond);//唤醒一批线程
int pthread_cond_signal(pthread_cond_t *cond);//唤醒某个线程

示例代码

#include <iostream>
#include <pthread.h>
#include <string>
#include <unistd.h>
using namespace std;

#define NUM 4
typedef void (*func_t)(const string name, pthread_mutex_t *pmtx, pthread_cond_t *pcond);
volatile bool quit = false;

class ThreadData
{
public:
    ThreadData(string &name, func_t func, pthread_mutex_t *pmtx, pthread_cond_t *pcond)
    : _name(name), _func(func), _pmtx(pmtx), _pcond(pcond)
    {
    }

public:
    string _name;
    func_t _func;
    pthread_mutex_t *_pmtx;
    pthread_cond_t *_pcond;
};

void func1(const string name, pthread_mutex_t *pmtx, pthread_cond_t *pcond)
{
    while(!quit)
    {
        pthread_mutex_lock(pmtx);
        pthread_cond_wait(pcond, pmtx);//线程等待
        cout << name << " running... -- 1" << endl;
        // sleep(1);
        pthread_mutex_unlock(pmtx);
    }
}

void func2(const string name, pthread_mutex_t *pmtx, pthread_cond_t *pcond)
{
    while(!quit)
    {
        pthread_mutex_lock(pmtx);
        pthread_cond_wait(pcond, pmtx);//线程等待
        cout << name << " running... -- 2" << endl;
        // sleep(1);
        pthread_mutex_unlock(pmtx);
    }
}

void func3(const string name, pthread_mutex_t *pmtx, pthread_cond_t *pcond)
{
    while(!quit)
    {
        pthread_mutex_lock(pmtx);
        pthread_cond_wait(pcond, pmtx);//线程等待
        cout << name << " running... -- 3" << endl;
        // sleep(1);
        pthread_mutex_unlock(pmtx);
    }
}

void func4(const string name, pthread_mutex_t *pmtx, pthread_cond_t *pcond)
{
    while(!quit)
    {
        pthread_mutex_lock(pmtx);
        pthread_cond_wait(pcond, pmtx);//线程等待
        cout << name << " running... -- 4" << endl;
        // sleep(1);
        pthread_mutex_unlock(pmtx);
    }
}


void* Entry(void* args)
{
    ThreadData* tmp = (ThreadData*)args;
    tmp->_func(tmp->_name, tmp->_pmtx, tmp->_pcond);
    delete tmp;
    return nullptr;
}

int main()
{
    pthread_mutex_t mtx;
    pthread_cond_t cond;
    pthread_mutex_init(&mtx, nullptr);
    pthread_cond_init(&cond, nullptr);

    pthread_t tid[NUM];
    func_t funcs[NUM] = {func1, func2, func3, func4};
    for (int i = 0; i < NUM; i++)
    {
        string name = "thread ";
        name += to_string(i + 1);
        ThreadData* td = new ThreadData(name, funcs[i], &mtx, &cond);
        pthread_create(tid + i, nullptr, Entry, (void*)td);
    }

    int cnt = 10;
    while(cnt)
    {
        cout << "resume thread run code..." << cnt-- << endl;
        pthread_cond_signal(&cond);
        // pthread_cond_broadcast(&cond);
        sleep(1);
    }

    cout << "ctrl done" << endl;
    quit = true;
    pthread_cond_broadcast(&cond);

    for(int i = 0; i < NUM; i++)
    {
        pthread_join(tid[i], nullptr);
        cout << "pthread: " << tid[i] << " quit" << endl; 
    }

    pthread_mutex_destroy(&mtx);
    pthread_cond_destroy(&cond);
    return 0;
}

结果演示

 

按照一定的顺序执行。

🔥解释pthread_cond_wait中的互斥量

条件等待是线程间同步的一种手段,如果只有一个线程,条件不满足,一直等下去也都不会满足,所以必须还要有一个线程通过某些操作来改变共享变量,使得不满足的条件变得满足,并且友好的通知在条件变量上等待的线程。但是条件不会无缘无故的满足,这必然会牵扯到共享数据的改变。共享数据属于临界资源,因此一定要用互斥锁来保护,没有互斥锁的保护就无法安全的获取和修改共享数据了。

按照上面的说法,我们转换成代码,必须先上锁,检测到条件不满足时,pthread_cond_wait会解锁,然后在条件变量上等待,直到条件满足时,pthread_cond_wait又会重新加锁。

进入pthread_cond_wait函数后,会去检测条件是否满足,如果不满足就把互斥量变为1(解锁),直到条件满足后(pthread_cond_wait返回)将互斥量恢复成原样。

条件变量的规范使用如下

//等待条件代码
pthread_mutex_lock(&mtx);
while(条件检测)
    pthread_cond_wait(&cond, &mtx);
//修改条件
pthread_mutex_unlock(&mtx);


//条件满足,唤醒线程代码
pthread_mutex_lock(&mtx);
//设置条件满足
pthread_cond_signal(&cond);
pthread_mutex_unlock(&mtx);

 

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

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

相关文章

ChatGPT3.5和ChatGPT4.0、ChatGPT4o对比

一、ChatGPT3.5、ChatGPT4.0、ChatGPT4o对比 目前ChatGPT有三个主要版本&#xff0c;分别是ChatGPT3.5、ChatGPT4.0、ChatGPT4o&#xff0c;这三个版本之间有什么差异呢&#xff1f; 对比项ChatGPT3.5ChatGPT4.0ChatGPT4o参数数量1750亿约1万亿未公开输入文本文本、图片文本、…

2+1退休模式解析(王老师同款)

互联网商业圈有比较多的商家老板认识王老师&#xff0c;他在全国多个地方开展他的课程&#xff0c;还是有比较多的商家去学习&#xff0c;其实看小编文章也能学到。下面分析一下王老师所讲的21退休模式。 21退休模式优势&#xff1a; 1、裂变快。在21退休模式中&#xff0c;每…

k8s常见故障--yaml文件检查没有问题 pod起不来(一直处于创建中)

故障信息 pod一直处于创建中 查看pod详细信息显示 kubectl describe pod 容器id文字 Events: Type Reason Age From Message Normal Scheduled 5m30s default-scheduler Successfully assigned default/nginx-server2-f97c6b9d5-d6dsp to worker02 Warning FailedCreatePod…

算法-找出N个数组的共同元素

一、代码与执行结果 财经新闻是大众了解金融事件的重要渠道&#xff0c;现有N位编辑&#xff0c;分别对K篇新闻进行专业的编辑与排版。需要您找出被这N位编辑共同编辑过的新闻&#xff0c;并根据这些新闻ID升序排列返回一个数组。 import random# 查找编辑共同处理的新闻id def…

U盘安装windows11详解教程

准备工作 准备好一个微PE启用U盘&#xff0c;将需要安装的win11镜像拷贝进去。详见教程制作PE启动U盘教程-CSDN博客 详细操作 1。首先我们需要将U盘插入电脑的USB接口。tips&#xff1a;尽量将U盘插到主机后置USB接口&#xff0c;不要将U盘插到前置USB接口或者延长线上&…

微型丝杆与滚珠丝杆性能差异与适用场景!

滚珠丝杆是工具机械和精密机械上最常使用的传动元件&#xff0c;其主要功能是将旋转运动转换成线性运动&#xff0c;或将扭矩转换成轴向反复作用力。同时兼具高精度、可逆性和高效率的特点。而微型丝杆是一种直径为0.5mm以下且线性误差在几微米以内&#xff0c;精度高、传动稳定…

LabVIEW与PLC的区别

LabVIEW和PLC是工业自动化领域中常见的两种控制和测控方案&#xff0c;各自有独特的优点和适用场景。本文将从多角度比较两者&#xff0c;帮助用户在选择控制系统时做出更明智的决策。 技术背景 LabVIEW LabVIEW是由National Instruments公司开发的图形化编程环境&#xff0…

win11家庭版安装docker

在 Windows 11 家庭版上安装 Docker 的详细教程 本文将详细介绍如何在 Windows 11 下安装 Docker。Docker 是一个开源的容器平台&#xff0c;可以轻松创建、部署和管理容器化应用程序。 1. 本电脑系统信息 在安装 Docker 之前&#xff0c;请确保您的 Windows 11 系统满足以下…

UE5中在地形中加入湖、河

系统水资产添加 前提步骤123 完成 前提 使用版本 UE5.0.3,使用插件为UE内置的Water和water Extras. 步骤 1 记得重启 2 增加地形&#xff0c;把<启用编辑图层>勾选 如果地形没有勾选上编辑图层&#xff0c;那么就会导致湖、河等水景象无法融入地形。 如果忘记勾选…

Redis 内存回收

文章目录 1. 过期key处理1.1 惰性删除1.2 周期删除 2. 内存淘汰策略 Redis 中数据过期策略采用定期删除惰性删除策略结合起来&#xff0c;以及采用淘汰策略来兜底。 定期删除策略&#xff1a;Redis 启用一个定时器定时监视所有的 key&#xff0c;判断key是否过期&#xff0c;过…

springcloudalibaba项目注册nacos,在nacos上修改配置项不生效问题

一、背景 之前的项目启动正常,后来发现springcloudalibaba的各版本匹配不正确,于是对项目中的springboot、springcloud、springcloudalibaba版本进行匹配升级,nacos1.4.2匹配的springboot、springcloud、springcloudalibaba版本与我的项目中的版本比较接近,于是我便重新安…

Vue的基础知识:v-model的原理,由:value与@input合写。

原理&#xff1a;v-model本质上是一个语法糖&#xff0c;比如应用在输入框上&#xff0c;就是value属性和input事件的合写。&#xff08;补充说明&#xff1a;语法糖就是语法的简写&#xff09; 作用&#xff1a;提供数据的双向绑定 1.数据变&#xff0c;视图&#xff08;也就…

MySQL 存储过程(一)

本篇主要介绍MySQL存储过程的相关内容 目录 一、什么是存储过程&#xff1f; 二、基本语法 创建存储过程 调用存储过程 查看存储过程 删除存储过程 三、变量 系统变量 用户自定义变量 局部变量 四、存储过程的参数 in out inout 一、什么是存储过程&#xff1f…

【React】classnames 优化类名控制

1. 介绍 classnames是一个简单的JS库&#xff0c;可以非常方便的通过条件动态的控制class类名的显示 ClassNames是一个用于有条件处理classname字符串连接的库 简单来说就是动态地去操作类名&#xff0c;把符合条件的类名粘在一起 现在的问题&#xff1a;字符串的拼接方式不…

uc/OS-III多任务程序

文章目录 一、实验内容二、实验步骤&#xff08;一&#xff09;基于STM32CubeMX建立工程&#xff08;二&#xff09;获取uc/OS-III源码&#xff08;三&#xff09;代码移植 三、修改mai.c文件四、实验现象 一、实验内容 学习嵌入式实时操作系统&#xff08;RTOS&#xff09;,以…

【python】python商业客户流失数据模型训练分析可视化(源码+数据集+课程论文)【独一无二】

&#x1f449;博__主&#x1f448;&#xff1a;米码收割机 &#x1f449;技__能&#x1f448;&#xff1a;C/Python语言 &#x1f449;公众号&#x1f448;&#xff1a;测试开发自动化【获取源码商业合作】 &#x1f449;荣__誉&#x1f448;&#xff1a;阿里云博客专家博主、5…

btstack协议栈实战篇--GAP Link Key Management

btstack协议栈---总目录-CSDN博客 目录 1.GAP 链接密钥逻辑 2.蓝牙逻辑 3.主应用程序设置 4.log信息 展示了如何遍历存储在 NVS 中的经典链接密钥&#xff0c;链接密钥是每个设备-设备绑定的。如果蓝牙控制器可以交换&#xff0c;例如在桌面系统上&#xff0c;则每个控制器都需…

Vue2入门(安装Vue、devtools,创建Vue)以及MVVM分层思想

文章目录 1.下载并安装Vue2.使用Vue2.1 创建Vue以及挂载Vue2.2 模板语句的数据来源&#xff1a;data2.3 template配置项详解2.4 Vue实例和容器的关系 3.安装devtools4.MVVM分层思想 1.下载并安装Vue &#xff08;1&#xff09;Vue是一个基于JavaScript&#xff08;JS&#xff…

数仓开发:如何计算投放效果?

背景介绍 业务介绍&#xff1a;用户是通过低价引流进来&#xff0c;然后通过复购购买高价商品&#xff0c;可以多次购买。低价商品和高价商品均可以退款&#xff0c;高价商品由于各种复杂的场景&#xff0c;可能会有多次退款。低价商品如果退款是全退&#xff0c;不存在多次退…

手猫助手Agent技术探索总结

随着LLM的发展&#xff0c;ChatGPT能力不断增强&#xff0c;AI不断有新的概念提出&#xff0c;一种衍生类型的应用AI Agent也借着这股春风开启了一波话题热度&#xff0c;各种初创公司&#xff0c;包括Open AI内部也都在密切关注着AI Agent领域的变化。阿里集团内的AI团队也有很…