【简记】Operating System——process synchronization

2023-11-28 20:50

本文主要是介绍【简记】Operating System——process synchronization,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

This memo is based on the course of Dr.Li with Operating System as the reference book.

本章内容:


6.1 背景

进程之间会共享数据,共享数据的过程中就可能出现数据不一致的情况。

如之前提到的生产者-消费者模型,在并发环境下,可能会出现线程(进程)不安全的情况,因为有一些非原子性的操作。例えば、变量counter,counter++和counter–这种看似不会出问题的语句,因为其底层实现的原因,会导致线程不安全的情况。

取指令,计算指令,写指令
这里写图片描述

上面的6个语句可能交叉执行,导致数据问题。


6.2 临界区问题

每个进程中有一个代码段称为临界区,在该区中进程可能会改变一个共同变量,更新一个表,写一个文件等。

这里写图片描述

解决临界区问题必须满足如下要求:

  • 互斥:同一时间内只有一个进程可在自己的临界区执行
  • 前进:如果没有进程在临界区内且有进程需要进入临界区,那么只有在那些不在剩余区内执行的进程可参加选择,且这种选择不能无限推迟
  • 有限等待:从一个进程做出进入临界区的请求,直到该请求允许为止。其他进程允许进入其临界区的次数有上限。

算法1(满足互斥,不满足前进)(why)

do{while(turn != i);ctrtical sectionturn = j;remainder section;}while(1)do{while(turn != j);ctrtical sectionturn = i;remainder section;}while(1)

算法2(满足互斥,不满足progress)(why)


进程pi
do{flag[i] = true;while(flag[j]);critical sectionflag[i] = false;remainder section}while(1)进程pj
do{flag[j] = true;while(flag[i]);critical sectionflag[j] = false;remainder section}while(1)

6.3 Peterson算法

两个进程P0和P1

两个进程间共享turn和flag

do{flag[i] = true;turn = j;while(flag[j]&&turn==j);临界区flag[i] = false;剩余区
}while(true)do{flag[j] = true;turn = i;while(flag[i]&&turn==i);临界区jflag[j] = false;剩余区
}while(true)

这里写图片描述


补充:Lamport面包房算法

n个进程的临界问题

该算法的基本思想源于顾客在面包店中购买面包时的排队原理. 顾客在进入面包店前, 首先抓一个号, 然后按照号码由小到大的次序依次进入面包店购买面包. 这里, 面包店发放的号码是由小到大的, 但是两个或两个以上的顾客却有可能得到相同的号码(使所抓号码不同需要互斥), 如果多个顾客抓到相同的号码, 则规定按照顾客名字的字典次序进行排序, 这里假定顾客是没有重名的. 在计算机系统中, 顾客就相当于进程, 每个进程有一个唯一的标识, 我们用P的下面加一个下标来表示. 例如: 对于 Pi和Pj, 如果有i < j, 则先为Pi服务, 即Pi先进入临界区

/*boolean choosing[n];表示进程是否在取号
int number[n];记录每个进程取到的号码
这些数据结构分别初始化为false0,为了方便,定义如下符号:
若a<c或a==c和b<d同时成立,(a,b)<(c,d)
即如果两个进程的排队号相同,则按进程的标识号进行比较
*
/
do
{
choosing[i] = true;
number[i] = max{number[0],number[1],…,number[n-1]}+1;//选号码,因为这里不是原子操作,所以可能会得到相同的号码
choosing[i] = false;
for(j = 0; j<n; j++)
{
while (choosing[j]); // 
while ((number[j] != 0) && (number[j],j)<(number[i],i)); // 当number[j]==0,说明该进程j并没有进临界区的请求,即可跳出while;当number[j]==number[i],两个进程的排队号相同,则进程号更大的会被留在while循环中
};
//临界区
number[i] = 0;
//其余部分
}while(1);
理解:第一个试图进入临界区的进程Pi在没有其它进程竞争时顺利进入其临界区。满足了有空就进的要求。当有竞争者Pk(i<k),且选的号码相同时,比较进程的名称,通过语句while ((number[j] != 0) && (number[j],j)<(number[i],i));来选择进入的进程。在Pi进程中j=i时,number[j]=number[i],且j=i所以(number[j],j)<(number[i],i)不成立,退出while语句。j==k时,number[k]=number[i],且k>i所以(number[j],j)<(number[i],i))不成立,退出while语句。对与其它非i和k的j值,number[j]!=0不成立,退出while语句。在Pk进程中j=i时,number[j]=number[k],且j<k,所以(number[j],j)<(number[i],i))成立,故进程Pk陷在该语句中,直到Pi退出临界区执行语句number[i]==0时才允许Pk进程进入其临界区(因为此时(number[j] != 0)这个条件不满足)。这里也有可能发生Pi退出临界区后继续获得CPU使用权并得到了一个新的排队号,但此时它所得到的排队号是max加1,所以Pi的排队号大于Pk,可以跳出while循环。所以满足了互斥和有限等待的要求。
为什么choosing数组是必要的?
https://www.zhihu.com/question/23336411/answer/44756007
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。可以想像这样一个例子: 现在排号已经排到了n,这个时候来了两个人i和j (i < j),他们都想领取n+1这个编号。如果去掉choosing,可能出现这样一种情况:虽然这两个人同时看到了现在的最大编号为n,也都想把自己的编号设置为n+1,但是j眼疾手快,抢在i之前去检查。i反应太迟钝了,以至于j检查到他的时候,他手上的编号纸还是空白的(number[i]=0),于是j顺理成章优哉游哉进店买面包了。这个时候慢吞吞的i终于在编号纸上写上了n+1,于是他也开始检查别人。当它检查到j的时候,发现他们两个的编号一样(number[i]=number[j]=n+1),而且自己的优先级还比j高(i<j),于是他也理直气壮地进店买面包了。一店不可容二主,于是杯具了。为什么choosing就可以阻止杯具的发生呢?我们穿越回j眼疾手快开始检查别人的时刻,当他检查到i的时候,发现choosing[i]=true(因为这个时候i还在慢吞吞没有写编号,没写编号choosing[i]就一直是true啦),所以j就不敢进店了,他必须等着choosing[i]被设成false。过了很久很久,i终于在自己的编号纸上写了编号n+1,也随之把choosing[i]设成了false。这个时候j高兴了,因为至少choosing[i]=false这道坎跨过去了。。。但是天不遂人愿,他随即发现虽然他们两个的编号一样,但是i的优先级比他的高(i<j),所以他又得乖乖等着。于是这个时候i逆袭的机会来了有木有!i火速检查别人,检查到j的时候发现choosing[j]=false,两个人的编号相同且自己的优先级更高,所以果断进去买面包。等i买完了出了店,他把自己的number[i]设成0,而这也正是j苦苦等待的。。。于是j也进店买了面包。至此,i和j一前一后买到了面包,很和谐很有爱,over.

6.4 硬件同步

之前的临界区问题产生的原因,都在于代码在执行的过程中会被CPU的调度所打断,所以会出现不同步的问题。

可以从软件和硬件的角度来解决,体现原子操作。

许多现代计算机系统提供了特殊硬件指令以允许原子地(不可中断地)检查和修改字的内容或交换两个字的内容。

下面的代码用TestAndSet解决了互斥,但是没有解决有限等待。(有可能某个线程一直得到CPU)全局布尔变量lock初始为false

// TestAndSet
boolean TestAndSet(boolean target){boolean rv = target;target = true;return rv
}
// 任意进程数量都可以满足
do{while(TestAndSet(lock));//ctrical section;lock = false;//remainder section;
}while(true);

swap函数,全局布尔变量lock为false,每个进程有一个局部boolean变量

do{key = true;while(key==true)swap(lock,key);//critical section;lock = false;//remainder section;
}while(true);

6.5 信号量

信号量是一个整数变量,除了初始化外,它只能通过两个标准原子操作:wait()和signal()来访问

当信号量是一个二进制数时,值只有0和1两种,有的系统也将二进制信号量称为互斥锁。
这里写图片描述
这里写图片描述
要求无法进入临界区的进程忙等待,这种类型的信号量也被称为自旋锁。自旋锁的优点是在等待锁时不进行上下文切换,在多处理器系统中,一个线程在处理器自旋时,另一个线程可以z在另一处理器上在其临界区执行。此种情况下信号量的值不会为负。

另一种信号量定义如下:
每个信号量都有一个整型值和一个进程链表,当一个进程必须等待信号量时,就加入到进程链表里。

wait(S){S--;if(S<0){add this process to waiting queue;block();}
}signal(S){S++;if(S<=0){remove a process P from the waiting queuewakeup(P) // 即上面的wait函数}
}

S的值可以小于0,绝对值为等待的线程数

Semaphore S;
do{wait(S);//critical section;signal(S)//remainder section;
}while(1);

===
多CPU下实现wait和signal的原子性

单CPU可以“关中”操作(关闭中断),多CPU要借助于锁


6.5 死锁和饥饿
这里写图片描述


6.6 经典同步问题

6.6.1 生产者消费者问题
6.6.2 读者写者问题
6.6.3 哲学家进餐问题


6.7 管程

这篇关于【简记】Operating System——process synchronization的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Partical System

创建"粒子系统物体"(点击菜单GameObject -> Create Other -> Particle System) 添加"粒子系统组件"(点击Component -> Effects  ->Particle System) 粒子系统检视面板  点击粒子系统检视面板的右上角的"+"来增加新的模块。(Show All Modules:显示全部) 初始化模块: •

小技巧绕过Sina Visitor System(新浪访客系统)

0x00 前言 一直以来,爬虫与反爬虫技术都时刻进行着博弈,而新浪微博作为一个数据大户更是在反爬虫上不遗余力。常规手段如验证码、封IP等等相信很多人都见识过…… 当然确实有需要的话可以通过新浪开放平台提供的API进行数据采集,但是普通开发者的权限比较低,限制也比较多。所以如果只是做一些简单的功能还是爬虫比较方便~ 应该是今年的早些时候,新浪引入了一个Sina Visitor Syst

System.getProperties().

Java.version Java 运行时环境版本 java.vendor Java 运行时环境供应商 java.vendor.url Java 供应商的 URL java.home Java 安装目录 java.vm.specification.version Java 虚拟机规范版本 java.vm.specification.vendor

12C 新特性,MOVE DATAFILE 在线移动 包括system, 附带改名 NID ,cdb_data_files视图坏了

ALTER DATABASE MOVE DATAFILE  可以改名 可以move file,全部一个命令。 resue 可以重用,keep好像不生效!!! system照移动不误-------- SQL> select file_name, status, online_status from dba_data_files where tablespace_name='SYSTEM'

Unity Post Process Unity后处理学习日志

Unity Post Process Unity后处理学习日志 在现代游戏开发中,后处理(Post Processing)技术已经成为提升游戏画面质量的关键工具。Unity的后处理栈(Post Processing Stack)是一个强大的插件,它允许开发者为游戏场景添加各种视觉效果,如景深、色彩校正、辉光、模糊等。这些效果不仅能够增强游戏的视觉吸引力,还能帮助传达特定的情感和氛围。 文档

android6/7 system打包脚本

1.android5打包system就是网站上常见的制作ROM必备的解包打包system脚本 指令如下:mkuserimg.sh -s out/target/product/$TARGET_PRODUCT/system out/target/product/$TARGET_PRODUCT/obj/PACKAGING/systemimage_intermediates/system.img

android打包解包boot.img,system.img

原帖地址:http://www.52pojie.cn/thread-488025-1-1.html 转载Mark一下,日后研究 最近工作需要对boot.img,system.img进行破解。顺便将心得分享一下。 我的工作环境是在linux下的。所以工具都是针对linux的。 boot.img破解相关工具: 1、split_boot    perl脚本 2、boot_i

MTK Android P/Q system/vendor/super快速打包

一、Android 新版本默认开启了动态分区,把system vendor  product等分区打包成一个super分区。这对于我们使用替换分区的方法来排查问题不是很方便,直接替换一个super也不知道到底是哪个部分导致的。所以我们需要自己制作super.img来缩小范围。下面讲讲如何快速生成system、vendor、super,以及vbmeta(校验image,不匹配可能会导致不开机) 二

Linux函数fcntl/system学习

本文针对项目中用到的几个函数进行详细分析,并尽可能的添加示例进行验证学习。比如fcntl/ioctl函数、system/exec函数、popen/pclose函数、mmap函数等。 重点参考了《UNP》和《Linux程序设计》第四版。 一、fcntl函数 fcntl函数可以改变或者查看已打开文件的性质。该函数的定义如下: #include <fcntl.h> int fcntl(

【UVA】11400-Lighting System Design(动态规划)

这道题感觉状态式不是很好推。。。 WA了好几次是因为排序的时候出问题了。 这道题出在线性结构里了,先说一下最长上升子序列吧。 dp[i]代表了以array[i]结尾的时候,最长子序列长度。 推导的时候,以起点递增的顺序进行推导。 #include<cstdio>#include<cstring>#include<iostream>#include<algorithm>#i