高编:线程(2)——同步与互斥

一、互斥

概念:
    互斥 ===》在多线程中对 临界资源 的 排他性访问。

    互斥机制 ===》互斥锁  ===》保证临界资源的访问控制。

    pthread_mutex_t   mutex;
    互斥锁类型            互斥锁变量 内核对象

框架:
     定义互斥锁 ==》初始化锁 ==》加锁 ==》解锁 ==》销毁
        ****                                           ***           ***

     1、定义

      pthread_mutex_t   mutex;

     2、初始化锁

        int pthread_mutex_init(pthread_mutex_t *mutex,const pthread_mutexattr_t *attr);
        功能:将已经定义好的互斥锁初始化。
        参数:mutex  要初始化的互斥锁
                   atrr  初始化的值,一般是NULL表示默认锁
        返回值:成功 0
                      失败 非零

    3、加锁

        int pthread_mutex_lock(pthread_mutex_t *mutex);
        功能:用指定的互斥锁开始加锁代码
                   加锁后的代码 到 解锁部分的代码 属于 原子操作(计算机中不可再分割的操作)
                   在加锁期间 其他进程/线程 都不能 操作该部分代码
                   如果该函数在执行的时候,mutex已经被其他部分使用代码阻塞

        参数: mutex 用来给代码加锁的互斥锁
        返回值:成功  0
                      失败  非零

     4、解锁

        int pthread_mutex_unlock(pthread_mutex_t *mutex);
        功能:将指定的互斥锁解锁。
                   解锁之后代码不再排他访问,一般加锁解锁同时出现。(尽量上锁解锁的代码短一些,保护并发性)
        参数:mutex  用来解锁的互斥锁
        返回值:成功  0
                      失败  非零

     5、销毁         

        int pthread_mutex_destroy(pthread_mutex_t *mutex);
         功能:使用互斥锁完毕后需要销毁互斥锁
         参数:mutex  要销毁的互斥锁
         返回值:成功  0
                       失败  非零

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <pthread.h>int a = 0;
pthread_mutex_t mutex;
void* th(void* arg)
{// pthread_mutex_lock(&mutex);int i =5000;while(i--){pthread_mutex_lock(&mutex);int tmp = a;printf("a is %d\n",tmp+1);a = tmp+1;pthread_mutex_unlock(&mutex); }// pthread_mutex_unlock(&mutex); return NULL;
}int main(int argc, char *argv[])
{pthread_t tid1,tid2;pthread_mutex_init(&mutex,NULL);pthread_create(&tid1,NULL,th,NULL);pthread_create(&tid2,NULL,th,NULL);pthread_join(tid1,NULL);pthread_join(tid2,NULL);pthread_mutex_destroy(&mutex);return 0;
}

 

6、trylock

        int pthread_mutex_trylock(pthread_mutex_t *mutex);
        功能:类似加锁函数效果,唯一区别就是不阻塞
        参数:mutex 用来加锁的互斥锁
        返回值:成功  0
                      失败  非零
                       E_AGAIN

 


综合练习:10个人去三个窗口办业务,办完一个人才能开始办下一个人

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <pthread.h>
pthread_mutex_t mutex;
int WIN = 3;
void* th(void* arg)
{while(1){pthread_mutex_lock(&mutex);if(WIN>0){WIN--;pthread_mutex_unlock(&mutex);printf("get win\n");sleep(rand()%5);printf("relese win\n");pthread_mutex_lock(&mutex);WIN++;pthread_mutex_unlock(&mutex);break;}else {pthread_mutex_unlock(&mutex);}}return NULL;
}int main(int argc, char *argv[])
{pthread_t tid[10];int i = 0 ;pthread_mutex_init(&mutex,NULL);for(i = 0 ;i<10;i++){pthread_create(&tid[i],NULL,th,NULL);}for(i = 0 ;i<10;i++){pthread_join(tid[i],NULL);}pthread_mutex_destroy(&mutex);return 0;
}

 二、同步

        同步 ===》有一定先后顺序的对资源的排他性访问。

    原因:互斥锁可以控制排他访问没有次序

    linux下的线程同步  ===》信号量机制 ===》semaphore.h   posix 
    sem_open();
    信号量的分类:
    1、无名信号量 ==》线程间通信
    2、有名信号量 ==》进程间通信

    框架:
    信号量的定义 ===》信号量的初始化 ===》信号量的P(申请资源)V(释放资源)操作 

===》信号量的销毁。

    semaphore 

1、信号量的定义 

        sem_t                sem;
       信号量的类型     信号量的变量

2、信号量的初始化

        int sem_init(sem_t *sem, int pshared, unsigned int value);
        功能:将已经定义好的信号量赋值。
        参数:sem 要初始化的信号量
                   pshared = 0 ;表示线程间使用信号量
                                 !=0 ;表示进程间使用信号量
                   value 信号量的初始值,一般无名信号量
                            都是二值信号量,0 1 
                            0 表示红灯,进程暂停阻塞
                            1 表示绿灯,进程可以通过执行
        返回值:成功  0
                      失败  -1;

3、信号量的PV 操作

       P ===》申请资源===》申请一个二值信号量 
       V ===》释放资源===》释放一个二值信号量

       P操作对应函数 ==》sem_wait();
       V操作对应函数 ==》sem_post();


    int sem_wait(sem_t *sem);//会阻塞
    功能:判断当前sem信号量是否有资源可用
              如果sem有资源(==1),则申请该资源,程序继续运行
              如果sem没有资源(==0),则线程阻塞等待,一旦有资源,则自动申请资源并继续运行程序。

     注意:sem 申请资源后会自动执行 sem = sem - 1;
     参数:sem 要判断的信号量资源
     返回值:成功 0 
                   失败 -1


     int sem_post(sem_t *sem);
    功能:函数可以将指定的sem信号量资源释放
              并默认执行,sem = sem + 1;
              线程在该函数上不会阻塞
    参数:sem 要释放资源的信号量
    返回值:成功 0
                  失败 -1;

4、信号量的销毁

        int sem_destroy(sem_t *sem);
       功能:使用完毕将指定的信号量销毁
       参数:sem要销毁的信号量
       返回值:成功 0
                     失败  -1;

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <pthread.h>
#include <semaphore.h>
sem_t sem_C,sem_M;
void* th1(void* arg)
{int i = 5;while(i--){sem_wait(&sem_C);printf("capper ");fflush(stdout);sem_post(&sem_M);}return NULL;
}
void* th2(void* arg)
{int i = 5;while(i--){sem_wait(&sem_M);printf("made in future\n");sleep(1);sem_post(&sem_C);}return NULL;
}int main(int argc, char *argv[])
{pthread_t tid1,tid2;sem_init(&sem_C,0,1);sem_init(&sem_M,0,0);pthread_create(&tid1,NULL,th1,NULL);pthread_create(&tid2,NULL,th2,NULL);pthread_join(tid1,NULL);pthread_join(tid2,NULL);sem_destroy(&sem_C);sem_destroy(&sem_M);return 0;
}


三、互斥锁和信号量的异同

同一时刻只有一个在走

1. 运行顺序--->互斥锁没有顺序信号量有顺序
2. 锁的个数--->互斥锁只有1个锁信号量一般大于2个锁(几个同步变量就几个锁)
3. 加锁解锁操作--->互斥锁同一个线程:同一个锁上锁解锁;信号量同一个线程不同锁:锁自己,释放下一个锁

四、死锁

1.产生死锁的原因

(1) 因为系统资源不足。
(2) 进程运行推进的顺序不合适。
(3) 资源分配不当等。
如果系统资源充足,进程的资源请求都能够得到满足,死锁出现的可能性就很低,否则
就会因争夺有限的资源而陷入死锁。其次,进程运行推进顺序与速度不同,也可能产生死锁。

2.产生死锁的四个必要条件(同时满足 产生死锁)

(1) 互斥条件:一个资源每次只能被一个进程使用。
(2) 请求与保持条件:一个进程因请求资源而阻塞时,对已获得的资源保持不放。
(3) 不剥夺条件:进程已获得的资源,在末使用完之前,不能强行剥夺。
(4) 循环等待条件:若干进程之间形成一种 头尾相接 的循环等待资源关系。
     

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

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

相关文章

STL的六大组件

一.总体概念 STL&#xff08;Standard Template Library&#xff0c;标准模板库&#xff09;是C标准库的一部分&#xff0c;提供了丰富且高效的数据结构和算法。STL主要由六大组件组成&#xff0c;它们是&#xff1a; 容器&#xff08;Containers&#xff09;&#xff1a;STL提…

1.(vue3.x+vite)实现卷帘效果

前端技术社区总目录(订阅之前请先查看该博客) 1:效果预览 2:代码编写 <template><div style="width

如何使用Maxscript访问C#类库?

本教程帮助你一步一步集成maxscript文档和C#类库&#xff0c;以便你能够在maxscript提供的相当有限的环境中访问dotnet框架的强大功能。这让你可以使用maxscript做一些功能更强大的事情&#xff0c;比如访问数据库、获取web部署的内容等等。 还是直接上教程实惠&#xff1a; …

昇思25天学习打卡营第6天|关于函数与神经网络梯度相关技术探讨

目录 Python 库及 MindSpore 相关模块和类的导入 函数与计算图 微分函数与梯度计算 Stop Gradient Auxiliary data 神经网络梯度计算 Python 库及 MindSpore 相关模块和类的导入 Python 中的 numpy 库被成功导入&#xff0c;并简称为 np。numpy 在科学计算领域应用广泛&#x…

SQLyog脚本无限试用重置脚本

文章目录 引言脚本(win)必要操作、说明 引言 SQLyog 需要po jie&#xff0c;但是网上的没看到很好使的&#xff0c;直接下的官方。能处理14天试用也是很ok的。 脚本(win) echo offREM SQLyog注册表key&#xff0c;可能跟你的不一样&#xff0c;如果不一样&#xff0c;请替换…

ai绘画一条作品变现1400+,怎么做一个赚钱的AI绘画账号?

大家都知道现在AI很火&#xff0c;变现的玩法也多种多样&#xff0c;但一说到AI&#xff0c;大家就下意识认为这东西离我太远&#xff0c;自己没有那么高学历&#xff0c;不会英文&#xff0c;不会用AI模型等。 其实Ai没有大家想象得那么难&#xff0c;尤其在AI绘画这块&#…

DNS访问百度

DNS&#xff0c;英文全称是 domain name system&#xff0c;域名解析系统&#xff0c;它的作用也很明确&#xff0c;就是域名和 IP 相互映射。 假设你要查询 baidu.com 的 IP 地址: 首先会查找浏览器的缓存,看看是否能找到 baidu.com 对应的IP地址&#xff0c;找到就直接返回&…

【第七节】C/C++排序算法

目录 前言 一、冒泡排序 二、选择排序 三、插入排序 四、希尔排序 五、归并排序 六、快速排序 七、 堆排序 八、计数排序 九、桶排序 十、基数排序 前言 排序算法可以大致分为两大类&#xff1a;比较类排序和非比较类排序。以下是这两大类中一些常见的排序算法示例&…

ChatGPT-4o医学应用、论文撰写、数据分析与可视化、机器学习建模、病例自动化处理、病情分析与诊断支持

2022年11月30日&#xff0c;可能将成为一个改变人类历史的日子——美国人工智能开发机构OpenAI推出了聊天机器人ChatGPT-3.5&#xff0c;将人工智能的发展推向了一个新的高度。2023年11月7日&#xff0c;OpenAI首届开发者大会被称为“科技界的春晚”&#xff0c;吸引了全球广大…

WPF布局控件

目录 Grid StackPanel WrapPanel DockPanel UniformGrid Canvas&InkCanvas Canvas InkCanvas Border Grid 属性 ShowGridLines&#xff1a;显示边线 ColumnDefinitions 列集合 表示有几列下面就写几个ColumnDefinition Width 宽&#xff1a;如果写具体数字则表…

科普文:一文搞懂jvm实战(一)Runtime实时监控jvm

概叙 Java Runtime 类是 Java 标准库中的关键类之一。它提供了对当前Java虚拟机&#xff08;JVM&#xff09;实例的访问和控制&#xff0c;允许程序动态地修改和管理运行时环境。 Java Runtime 是Java虚拟机&#xff08;JVM&#xff09;的一个实例&#xff0c;代表了正在执行Ja…

Spring Boot 实现 AOP 动态热插拔功能并附DEMO源码

&#x1f604; 19年之后由于某些原因断更了三年&#xff0c;23年重新扬帆起航&#xff0c;推出更多优质博文&#xff0c;希望大家多多支持&#xff5e; &#x1f337; 古之立大事者&#xff0c;不惟有超世之才&#xff0c;亦必有坚忍不拔之志 &#x1f390; 个人CSND主页——Mi…

python中的包和模块

目录 一、包与模块 二、第三方包的安装 2.1 pip install 2.2使用 curl 管道 2.3其他安装方法 三、导入单元的构成 3.1pip的使用 四、模块的缓存 一、包与模块 Python 中除了函数库以外&#xff0c;还有非常多且优秀的第三方库、包、模块。 模块Module&#xff1a;以…

【linux网络(七)】数据链路层详解

&#x1f493;博主CSDN主页:杭电码农-NEO&#x1f493;   ⏩专栏分类:Linux从入门到精通⏪   &#x1f69a;代码仓库:NEO的学习日记&#x1f69a;   &#x1f339;关注我&#x1faf5;带你学更多操作系统知识   &#x1f51d;&#x1f51d; Linux网络 1. 前言2. 认识MAC…

【计算机毕业设计】061互助学习微信小程序

&#x1f64a;作者简介&#xff1a;拥有多年开发工作经验&#xff0c;分享技术代码帮助学生学习&#xff0c;独立完成自己的项目或者毕业设计。 代码可以私聊博主获取。&#x1f339;赠送计算机毕业设计600个选题excel文件&#xff0c;帮助大学选题。赠送开题报告模板&#xff…

北京网站建设怎么开始做

北京作为中国的首都&#xff0c;拥有众多的企业和机构&#xff0c;网站建设不仅是一种宣传和推广的手段&#xff0c;更是企业发展的必备工具。但是对于很多企业来说&#xff0c;网站建设是一个相对陌生的领域&#xff0c;不知道从哪里开始。今天我们就来谈一谈北京网站建设的步…

Dockerhub无法拉取镜像配置阿里镜像加速器

打开阿里镜像加速地址&#xff1a; https://cr.console.aliyun.com/cn-hangzhou/instances/mirrors 根据平台类型按照对应方式进行配置&#xff1a;Dokcer Desktop是在右上角点开配置 找到Docker Engine 进行设置JSON结构&#xff1a; 记得要重启Docker服务才会生效&#xff01…

深度学习笔记: 最详尽解释预测系统的分类指标(精确率、召回率和 F1 值)

欢迎收藏Star我的Machine Learning Blog:https://github.com/purepisces/Wenqing-Machine_Learning_Blog。如果收藏star, 有问题可以随时与我交流, 谢谢大家&#xff01; 预测系统的分类指标(精确率、召回率和 F1 值) 简介 让我们来谈谈预测系统的分类指标以及对精确率、召回…

SpringSecurity6 | 基于数据库实现登录认证

SpringSecurity6 | 基于数据库认证 ✅作者简介:大家好,我是Leo,热爱Java后端开发者,一个想要与大家共同进步的男人😉😉 🍎个人主页:Leo的博客 💞当前专栏: 循序渐进学SpringSecurity6 ✨特色专栏: MySQL学习 🥭本文内容: SpringSecurity6 | 基于数据库实现登…

数据资产的价值变现及管理规划(AMT企源)

从数据资源到数据资产之数据资产的价值变现及管理规划 题记 本文旨在探讨数据资产发展途径、数据产权及价值变现的服务流程和路径&#xff0c;并对数据资产管理平台框架、数据资产管理实施规划做出初步解读&#xff0c;以期为数据资产管理提供有益的思路和方案。 本次推出《从…