进程死锁算法——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

相关文章

SQL Server数据库死锁处理超详细攻略

《SQLServer数据库死锁处理超详细攻略》SQLServer作为主流数据库管理系统,在高并发场景下可能面临死锁问题,影响系统性能和稳定性,这篇文章主要给大家介绍了关于SQLServer数据库死... 目录一、引言二、查询 Sqlserver 中造成死锁的 SPID三、用内置函数查询执行信息1. sp_w

Windows的CMD窗口如何查看并杀死nginx进程

《Windows的CMD窗口如何查看并杀死nginx进程》:本文主要介绍Windows的CMD窗口如何查看并杀死nginx进程问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地... 目录Windows的CMD窗口查看并杀死nginx进程开启nginx查看nginx进程停止nginx服务

Java死锁问题解决方案及示例详解

《Java死锁问题解决方案及示例详解》死锁是指两个或多个线程因争夺资源而相互等待,导致所有线程都无法继续执行的一种状态,本文给大家详细介绍了Java死锁问题解决方案详解及实践样例,需要的朋友可以参考下... 目录1、简述死锁的四个必要条件:2、死锁示例代码3、如何检测死锁?3.1 使用 jstack3.2

Java进程CPU使用率过高排查步骤详细讲解

《Java进程CPU使用率过高排查步骤详细讲解》:本文主要介绍Java进程CPU使用率过高排查的相关资料,针对Java进程CPU使用率高的问题,我们可以遵循以下步骤进行排查和优化,文中通过代码介绍... 目录前言一、初步定位问题1.1 确认进程状态1.2 确定Java进程ID1.3 快速生成线程堆栈二、分析

Python多进程、多线程、协程典型示例解析(最新推荐)

《Python多进程、多线程、协程典型示例解析(最新推荐)》:本文主要介绍Python多进程、多线程、协程典型示例解析(最新推荐),本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定... 目录一、multiprocessing(多进程)1. 模块简介2. 案例详解:并行计算平方和3. 实现逻

C#通过进程调用外部应用的实现示例

《C#通过进程调用外部应用的实现示例》本文主要介绍了C#通过进程调用外部应用的实现示例,以WINFORM应用程序为例,在C#应用程序中调用PYTHON程序,具有一定的参考价值,感兴趣的可以了解一下... 目录窗口程序类进程信息类 系统设置类 以WINFORM应用程序为例,在C#应用程序中调用python程序

使用雪花算法产生id导致前端精度缺失问题解决方案

《使用雪花算法产生id导致前端精度缺失问题解决方案》雪花算法由Twitter提出,设计目的是生成唯一的、递增的ID,下面:本文主要介绍使用雪花算法产生id导致前端精度缺失问题的解决方案,文中通过代... 目录一、问题根源二、解决方案1. 全局配置Jackson序列化规则2. 实体类必须使用Long封装类3.

Springboot实现推荐系统的协同过滤算法

《Springboot实现推荐系统的协同过滤算法》协同过滤算法是一种在推荐系统中广泛使用的算法,用于预测用户对物品(如商品、电影、音乐等)的偏好,从而实现个性化推荐,下面给大家介绍Springboot... 目录前言基本原理 算法分类 计算方法应用场景 代码实现 前言协同过滤算法(Collaborativ

Python如何精准判断某个进程是否在运行

《Python如何精准判断某个进程是否在运行》这篇文章主要为大家详细介绍了Python如何精准判断某个进程是否在运行,本文为大家整理了3种方法并进行了对比,有需要的小伙伴可以跟随小编一起学习一下... 目录一、为什么需要判断进程是否存在二、方法1:用psutil库(推荐)三、方法2:用os.system调用

Mysql如何解决死锁问题

《Mysql如何解决死锁问题》:本文主要介绍Mysql如何解决死锁问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录【一】mysql中锁分类和加锁情况【1】按锁的粒度分类全局锁表级锁行级锁【2】按锁的模式分类【二】加锁方式的影响因素【三】Mysql的死锁情况【1