进程死锁算法——Peterson与Dekker

2024-01-13 02:58

本文主要是介绍进程死锁算法——Peterson与Dekker,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

进来Bear正在学习巩固并行的基础知识,所以写下这篇基础的有关并行算法的文章。

在讲述两个算法之前,需要明确一些概念性的问题,

Race Condition(竞争条件),Situations  like  this,  where  two  or  more processes  are  reading or writing some shared data and the final result depends on who runs precisely when, are called race conditions.多个进程(线程)读写共享区域(共享文件、共享变量、共享内存等)时,最后的结果依赖于他们执行的相对时间。

Critical Regions(关键区域),That part of the program where the shared memory is accessed.在 程序 中,访问共享内存的部分。

Mutual exclusion(互斥), that is, some way of making sure that if one process is using a shared  variable or file, the other processes will be excluded from doing the same thing.指在一个进程在使用共享区域时,防止另外的进程做同样的事情。

同样,需要四个条件来找到一个好的解决方案,实现进程(线程)之间的互斥:

Dekker算法与Peterson算法就是用来实现进程或者线程之间的互斥。

Dekker算法:(参考了百度百科上面的Dekker算法解析,还是挺易懂的)

Dekker算法可以用于控制两个进程(线程)之间的同步,如下实现的功能就是专门用于线程的同步:

其中,flag[2]用来表示是否想要使用关键区,turn用来表示具有访问权限的进程ID。( 重点看注释,通过注释,挺好理解的哟~ )

#include<stdio.h>
#include<stdlib.h>
#include<pthread.h>
#define true 1
#define false 0
typedef int bool;
bool flag[2];
int turn;
void visit(int num)
{sleep(1);printf("P%d is visting\n",num);
}
void P0()
{while(true){flag[0] = true;//P0想使用关键区。while(flag[1])//检查P1是不是也想用?{if(turn == 1)//如果P1想用,则查看P1是否具有访问权限?{flag[0] = false;//如果有,则P0放弃。while(turn == 1);//检查turn是否属于P1。flag[0] = true;//P0想使用。}}visit(0); //访问Critical Partition。turn = 1; //访问完成,将权限给P1。flag[0] = false;//P0结束使用。}
}
void P1()
{while(true){flag[1] = true; //P1想使用关键区。while(flag[0]) //检查P0是不是也想用?{if(turn == 0) //如果P0想用,则查看P0是否具有访问权限?{flag[1] = false; //如果有,则P1放弃。while(turn == 0); //检查turn是否属于P1。flag[1] = true; // P1想使用。}}visit(1); //访问Critical Partition。turn = 0; //访问完成,将权限给P0。flag[1] = false; //P1结束使用。}
}
void main()
{pthread_t t1,t2;flag[0] = flag[1] = false;turn = 0;int err;err =  pthread_create(&t1,NULL,(void*)P0,NULL);if(err != 0) exit(-1);err = pthread_create(&t2,NULL,(void*)P1,NULL);if(err != 0 ) exit(-1);pthread_join(t1,NULL);pthread_join(t2,NULL);exit(0);
}

如上所示,如果 flag数组和turn都没有符合使用关键区的条件 的时候,是不可能进入关键区的。

Peterson算法:

Peterson算法也是保证两个进程(线程)实现互斥的方法,比之前的Dekker算法更加简单,他同样提供了两个变量,保证进程不进入关键区,一个是flag[2],一个是turn,两者的表达意思也类似,flag数组表示能否有权限使用关键区,turn是指有访问权限的进线程ID。( 注释很重要,帮助你理解 )

#include<stdio.h>
#include<stdlib.h>
#include<pthread.h>
#define true 1
#define false 0
typedef int bool;
bool flag[2];
int turn;
void procedure0()
{while(true){flag[0] = true;turn = 1;while(flag[1] && turn == 1)//退出while循环的条件就是,要么另一个线程
//不想要使用关键区,要么此线程拥有访问权限。{sleep(1);printf("procedure0 is waiting!\n");}//critical sectionflag[0] = false;}
}
void procedure1()
{while(true){flag[1] = true;turn = 0;while(flag[0] && turn == 0){sleep(1);printf("procedure1 is waiting!\n");}//critical sectionflag[1] = false;}
}
void main()
{pthread_t t1,t2;flag[0] = flag[1] = false;int err;turn = 0;err =  pthread_create(&t1,NULL,(void*)procedure0,NULL);if(err != 0) exit(-1);err = pthread_create(&t2,NULL,(void*)procedure1,NULL);if(err != 0 ) exit(-1);pthread_join(t1,NULL);pthread_join(t2,NULL);exit(0);
}

Bear将turn的赋值放在while循环的后面,然后main函数中赋初值,也是可行的。

这篇关于进程死锁算法——Peterson与Dekker的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/600020

相关文章

Python中的随机森林算法与实战

《Python中的随机森林算法与实战》本文详细介绍了随机森林算法,包括其原理、实现步骤、分类和回归案例,并讨论了其优点和缺点,通过面向对象编程实现了一个简单的随机森林模型,并应用于鸢尾花分类和波士顿房... 目录1、随机森林算法概述2、随机森林的原理3、实现步骤4、分类案例:使用随机森林预测鸢尾花品种4.1

python多进程实现数据共享的示例代码

《python多进程实现数据共享的示例代码》本文介绍了Python中多进程实现数据共享的方法,包括使用multiprocessing模块和manager模块这两种方法,具有一定的参考价值,感兴趣的可以... 目录背景进程、进程创建进程间通信 进程间共享数据共享list实践背景 安卓ui自动化框架,使用的是

C#如何优雅地取消进程的执行之Cancellation详解

《C#如何优雅地取消进程的执行之Cancellation详解》本文介绍了.NET框架中的取消协作模型,包括CancellationToken的使用、取消请求的发送和接收、以及如何处理取消事件... 目录概述与取消线程相关的类型代码举例操作取消vs对象取消监听并响应取消请求轮询监听通过回调注册进行监听使用Wa

不懂推荐算法也能设计推荐系统

本文以商业化应用推荐为例,告诉我们不懂推荐算法的产品,也能从产品侧出发, 设计出一款不错的推荐系统。 相信很多新手产品,看到算法二字,多是懵圈的。 什么排序算法、最短路径等都是相对传统的算法(注:传统是指科班出身的产品都会接触过)。但对于推荐算法,多数产品对着网上搜到的资源,都会无从下手。特别当某些推荐算法 和 “AI”扯上关系后,更是加大了理解的难度。 但,不了解推荐算法,就无法做推荐系

康拓展开(hash算法中会用到)

康拓展开是一个全排列到一个自然数的双射(也就是某个全排列与某个自然数一一对应) 公式: X=a[n]*(n-1)!+a[n-1]*(n-2)!+...+a[i]*(i-1)!+...+a[1]*0! 其中,a[i]为整数,并且0<=a[i]<i,1<=i<=n。(a[i]在不同应用中的含义不同); 典型应用: 计算当前排列在所有由小到大全排列中的顺序,也就是说求当前排列是第

csu 1446 Problem J Modified LCS (扩展欧几里得算法的简单应用)

这是一道扩展欧几里得算法的简单应用题,这题是在湖南多校训练赛中队友ac的一道题,在比赛之后请教了队友,然后自己把它a掉 这也是自己独自做扩展欧几里得算法的题目 题意:把题意转变下就变成了:求d1*x - d2*y = f2 - f1的解,很明显用exgcd来解 下面介绍一下exgcd的一些知识点:求ax + by = c的解 一、首先求ax + by = gcd(a,b)的解 这个

综合安防管理平台LntonAIServer视频监控汇聚抖动检测算法优势

LntonAIServer视频质量诊断功能中的抖动检测是一个专门针对视频稳定性进行分析的功能。抖动通常是指视频帧之间的不必要运动,这种运动可能是由于摄像机的移动、传输中的错误或编解码问题导致的。抖动检测对于确保视频内容的平滑性和观看体验至关重要。 优势 1. 提高图像质量 - 清晰度提升:减少抖动,提高图像的清晰度和细节表现力,使得监控画面更加真实可信。 - 细节增强:在低光条件下,抖

【数据结构】——原来排序算法搞懂这些就行,轻松拿捏

前言:快速排序的实现最重要的是找基准值,下面让我们来了解如何实现找基准值 基准值的注释:在快排的过程中,每一次我们要取一个元素作为枢纽值,以这个数字来将序列划分为两部分。 在此我们采用三数取中法,也就是取左端、中间、右端三个数,然后进行排序,将中间数作为枢纽值。 快速排序实现主框架: //快速排序 void QuickSort(int* arr, int left, int rig

poj 3974 and hdu 3068 最长回文串的O(n)解法(Manacher算法)

求一段字符串中的最长回文串。 因为数据量比较大,用原来的O(n^2)会爆。 小白上的O(n^2)解法代码:TLE啦~ #include<stdio.h>#include<string.h>const int Maxn = 1000000;char s[Maxn];int main(){char e[] = {"END"};while(scanf("%s", s) != EO

秋招最新大模型算法面试,熬夜都要肝完它

💥大家在面试大模型LLM这个板块的时候,不知道面试完会不会复盘、总结,做笔记的习惯,这份大模型算法岗面试八股笔记也帮助不少人拿到过offer ✨对于面试大模型算法工程师会有一定的帮助,都附有完整答案,熬夜也要看完,祝大家一臂之力 这份《大模型算法工程师面试题》已经上传CSDN,还有完整版的大模型 AI 学习资料,朋友们如果需要可以微信扫描下方CSDN官方认证二维码免费领取【保证100%免费