线程同步互斥机制

一、重要概念

临界资源:
多线程执行流共享的资源就叫做临界资源

临界区:
每个线程内部,访问临界资源的代码,就叫做临界区

互斥:
任何时刻,互斥保证有且只有一个执行流进入临界区,访问临界资源

原子性:
不会被任何调度机制打断的操作,该操作只有两态,完成和未完成

可重入:
重入:同一个函数被不同的执行流调用,当前一个流程还没有执行完,就有其他的执行流再次进入,我们称之为重入。一个函数在重入的情况下,运行结果仍正确,则该函数被称为可重入函数,否则,是不可重入函数。

死锁:
死锁是一种永久等待的状态,产生的原因是线程/线程互相申请被其他线程占用且不释放的资源

产生死锁的四个必要条件:
互斥:
一个资源每次只能被一个执行流使用
请求与保持条件:
一个执行流因请求资源而阻塞时,对已获得的资源保持占有
不剥夺条件:
一个执行流已获得的资源不能被强行剥夺
循环等待条件:
几个执行流之间形成一种头尾相接的循环等待资源的情况
死锁的解决方法:
解决死锁的方法可以从多个角度去分析,一般的情况下,有预防,避免,检测和解除四种。

饥饿:
如果一个线程因为处理器时间全部被其他线程抢走而得不到处理器运行时间,这种状态被称之为饥饿,一般是由高优先级线程吞噬所有的低优先级线程的处理器时间引起的

同步:
在保证数据安全的前提下,让线程能够按照某种特定的顺序访问临界资源,从而有效避免饥饿问题,叫做同步

二、线程安全

线程安全:
拥有共享数据的多个线程并行执行时,通过同步机制保证各个线程都可以正确执行,不出现数据污染就叫做线程安全。

三、互斥量

互斥量就是互斥锁,作用是确保同一时间只有一个线程访问共享资源。某个线程在访问共享资源前要加锁,加锁后其他试图访问资源和加锁的线程就会被阻塞,直到当前线程释放互斥锁。释放之后第一个变为运行状态的线程就可以对互斥量加锁然后访问共享资源了。

Linux通过lock和unlock达到上述目的。

在这里插入图片描述

1.互斥量的原理

在这里插入图片描述
加锁实现原理:
互斥量默认为1,执行xchgb后,只有一个线程寄存器的值为1,可以进入临界区。
互斥量此时为0,其他线程寄存器的值为0,即使其他线程寄存器和互斥量交换数值,寄存器里的值也是0,无法进入临界区。
线程寄存器数值为1的线程执行对应的代码后执行goto lock,线程寄存器的值置为0。

释放锁实现原理:
互斥量置1即可,其他线程可通过执行xchgb语句将线程寄存器的值置1后,进入临界区。

2.互斥量接口

初始化互斥量

静态初始化:
如果互斥锁 mutex 是静态分配的(定义在全局,或加static关键字修饰),可以直接用宏来初始化:

   pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;

动态初始化:

    int pthread_mutex_init(&mutex, NULL);   //mutex为定义的互斥量

销毁互斥量

    int pthread_mutex_destroy(pthread_mutex_t *mutex)

注意:
使用 PTHREAD_ MUTEX_ INITIALIZER 初始化的互斥量不需要手动销毁
不要销毁一个已经加锁的互斥量
已经销毁的互斥量,要确保后面不会有线程再尝试加锁

互斥量加锁解锁

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

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

加锁时的情景:

互斥量处于未锁状态,该函数会将互斥量锁定,同时返回成功
其他线程已经锁定互斥量,或者存在其他线程同时申请互斥量,但没有竞争到互斥量,那么
pthread_ lock会陷入阻塞(执行流被挂起),等待互斥量解锁。

四、条件变量

条件变量:
条件变量通常和互斥锁一起使用时,允许线程以无竞争的方式等待特定条件发生
条件变量的两个动作:
  1. 条件不满足: 阻塞线程
  2. 条件满足: 通知阻塞的线程开始工作
  
例如:
对于一个生产者消费者问题,如果没有产品而消费线程使用不断查询来判断是否有产品将大量的浪费CPU时间。如果没有产品时消费者线程可以先阻塞,等待有产品时被唤醒进行消费。

1.条件变量接口

初始化条件变量:

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

参数:cond:要初始化的条件变量
attr:一般设为NULL

销毁条件变量

int pthread_cond_destroy(pthread_cond_t *cond)

等待

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

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

这里参数传入互斥量是因为,如果lock后进行等待,还没有unlock,其他线程无法获取互斥量,会造成死锁。传入互斥量后,等待时自动释放互斥量,被唤醒时自动获取互斥量。

唤醒

  int pthread_cond_broadcast(pthread_cond_t *cond);  //唤醒所有该条件变量下挂起的线程
  int pthread_cond_signal(pthread_cond_t *cond);     //唤醒一个该条件变量下挂起的线程

2.伪唤醒

在线程被唤醒过程中,如果锁被其他线程抢占执行,等持锁线程执行完后,被唤醒线程获得锁执行,就有可能造成临界资源被过度消费为负数的现象(在生产者消费者模式中)。

避免伪唤醒:
当我们在线程唤醒后,再加一次对临界资源的判断,就能有效规避此问题,所以关于临界资源的判断语句不使用if而使用while

五、信号量

1.理解信号量

如果信号量是一个任意的整数,通常被称为计数信号量(一般信号量)
如果信号量只有二进制的0或1,称为二进制信号量,在linux系统中,二进制信号量又称互斥锁

信号量S代表临界资源的个数

信号量存在两种操作,V操作与P操作,V操作会增加信号量 S的数值,P操作会减少它。

具体的运作方式如下:

  1. 初始化,给与它一个非负数的整数值。

  2. 运行 P(wait()),信号量S的值将被减少。企图进入临界区的进程,需要先执行P操作。当信号量S减为负值时,进程会被挡住,不能继续;当信号量S不为负值时,进程可以获准进入临界区。

  3. 运行 V(signal()),信号量S的值会被增加。即将离开临界区的进程,将会执行V操作。当信号量S不为负值时,先前被挡住的其他进程,将可获准进入临界区。

此部分参自🔗

2.信号量接口

初始化信号量

#include <semaphore.h> 
  int sem_init(sem_t *sem, int pshared, unsigned int value); 

参数:
pshared:0表示线程间共享,非零表示进程间共享
value:信号量初始值

销毁信号量

   int sem_destroy(sem_t *sem); 

等待(P)

   int sem_wait(sem_t *sem); 

功能:会将信号量的值减1

唤醒(V)

   int sem_post(sem_t *sem);

功能:表示资源使用完毕,可以归还资源了,将信号量值加1。

  • 8
    点赞
  • 16
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 8
    评论
评论 8
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

江南无故人

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值