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

相关文章

怎样通过分析GC日志来定位Java进程的内存问题

《怎样通过分析GC日志来定位Java进程的内存问题》:本文主要介绍怎样通过分析GC日志来定位Java进程的内存问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、GC 日志基础配置1. 启用详细 GC 日志2. 不同收集器的日志格式二、关键指标与分析维度1.

Java进程异常故障定位及排查过程

《Java进程异常故障定位及排查过程》:本文主要介绍Java进程异常故障定位及排查过程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、故障发现与初步判断1. 监控系统告警2. 日志初步分析二、核心排查工具与步骤1. 进程状态检查2. CPU 飙升问题3. 内存

Java中的雪花算法Snowflake解析与实践技巧

《Java中的雪花算法Snowflake解析与实践技巧》本文解析了雪花算法的原理、Java实现及生产实践,涵盖ID结构、位运算技巧、时钟回拨处理、WorkerId分配等关键点,并探讨了百度UidGen... 目录一、雪花算法核心原理1.1 算法起源1.2 ID结构详解1.3 核心特性二、Java实现解析2.

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.