页面置换算法
页面置换算法实验报告实验心得
页面置换算法实验报告实验心得
页面置换算法是操作系统中用来管理内存的一种重要算法。
在本次实验中,我们通过模拟内存的分配和释放过程,探索了三种典型的页面置换算法:FIFO(先进先出)、LRU(最近最少使用)和OPT(最优置换)。
在实验过程中,我发现FIFO算法虽然简单易懂,但容易产生“抖动”现象,即容易出现频繁的页面置换,导致系统效率低下。
LRU算法则能够有效避免抖动现象,但需要记录每个页面最近一次的使用时间,算法实现较为复杂。
OPT算法是一种理论上的最优算法,但由于需要预测未来的页面使用情况,实际中难以实现。
通过对三种算法的实验数据分析,我发现在不同的内存使用情况下,不同的页面置换算法表现也不同。
例如在内存使用较少的情况下,FIFO算法的效率可能会更高,但在内存使用较多的情况下,LRU算法则能够更好地发挥作用。
因此,在实际应用中,需要根据实际情况选择合适的页面置换算法。
总之,本次页面置换算法的实验让我更加深入地了解了操作系统中内存管理的相关知识,也加深了我对算法选择的理解和实际应用的思考。
lru 页面置换算法
LRU 页面置换算法1. 简介LRU(Least Recently Used)页面置换算法是一种常用的操作系统内存管理算法,用于在内存不足时决定哪些页面应该被置换出去以腾出空间给新的页面。
LRU算法基于一个简单的原则:最近最少使用的页面应该被置换。
在计算机系统中,内存是有限的资源,而运行程序所需的内存可能超过可用内存大小。
当系统发现没有足够的空闲内存来加载新页面时,就需要选择一些已经在内存中的页面进行替换。
LRU算法就是为了解决这个问题而设计的。
2. 原理LRU算法基于一个简单的思想:如果一个页面最近被访问过,那么它将来可能会再次被访问。
相反,如果一个页面很久没有被访问过,那么它将来可能不会再次被访问。
根据这个思想,LRU算法将最近最少使用的页面置换出去。
具体实现上,可以使用一个数据结构来记录每个页面最近一次被访问的时间戳。
当需要替换一页时,选择时间戳最早(即最久未访问)的页面进行替换即可。
3. 实现方式LRU算法的实现可以基于多种数据结构,下面介绍两种常见的实现方式。
3.1 使用链表一种简单的实现方式是使用一个双向链表来记录页面的访问顺序。
链表头部表示最近访问过的页面,链表尾部表示最久未被访问过的页面。
每当一个页面被访问时,将其从原位置移动到链表头部。
当需要替换一页时,选择链表尾部的页面进行替换。
这种实现方式的时间复杂度为O(1),但空间复杂度较高,为O(n),其中n为内存中可用页面数。
class Node:def __init__(self, key, value):self.key = keyself.value = valueself.prev = Noneself.next = Noneclass LRUCache:def __init__(self, capacity):self.capacity = capacityself.cache = {}self.head = Node(0, 0)self.tail = Node(0, 0)self.head.next = self.tailself.tail.prev = self.headdef get(self, key):if key in self.cache:node = self.cache[key]self._remove(node)self._add(node)return node.valueelse:return -1def put(self, key, value):if key in self.cache:node = self.cache[key]node.value = valueself._remove(node)self._add(node)else:if len(self.cache) >= self.capacity:del self.cache[self.tail.prev.key] self._remove(self.tail.prev)node = Node(key, value)self.cache[key] = nodeself._add(node)def _remove(self, node):prev = node.prevnext = node.nextprev.next = nextnext.prev = prevdef _add(self, node):head_next = self.head.nextself.head.next = nodenode.prev = self.headnode.next = head_nexthead_next.prev = node3.2 使用哈希表和双向链表另一种实现方式是使用一个哈希表和一个双向链表。
页式虚拟存储管理FIFO、LRU和OPT页面置换算法
目录1 需求分析 (2)1.1 目的和要求 (2)1.2 研究内容 (2)2 概要设计 (2)2.1 FIFO算法 (3)2.2 LRU算法 (3)2.3 OPT算法 (3)2.4 输入新的页面引用串 (3)3 详细设计 (4)3.1 FIFO(先进先出)页面置换算法: (4)3.2 LRU(最近最久未使用)置换算法: (4)3.3 OPT(最优页)置换算法 (4)4 测试 (5)5 运行结果 (5)6 课程设计总结 (9)7 参考文献 (10)8 附录:源程序清单 (10)1 需求分析1.1 目的和要求在熟练掌握计算机虚拟存储技术的原理的基础上,利用一种程序设计语言模拟实现几种置换算法,一方面加深对原理的理解,另一方面提高学生通过编程根据已有原理解决实际问题的能力,为学生将来进行系统软件开发和针对实际问题提出高效的软件解决方案打下基础。
1.2 研究内容模拟实现页式虚拟存储管理的三种页面置换算法(FIFO(先进先出)、LRU (最近最久未使用)和OPT(最长时间不使用)),并通过比较性能得出结论。
前提:(1)页面分配采用固定分配局部置换。
(2)作业的页面走向和分得的物理块数预先指定。
可以从键盘输入也可以从文件读入。
(3)置换算法的置换过程输出可以在显示器上也可以存放在文件中,但必须清晰可读,便于检验。
2 概要设计本程序主要划分为4个功能模块,分别是应用FIFO算法、应用LRU算法、应用OPT算法和页面引用串的插入。
1.1各模块之间的结构图2.1 FIFO 算法该模块的主要功能是对相应页面引用串进行处理,输出经过FIFO 算法处理之后的结果。
2.2 LRU 算法该模块的主要功功能是对相应的页面引用串进行处理,输出经过LRU 算法处理之后的结果。
2.3 OPT 算法该模块的主要功功能是对相应的页面引用串进行处理,输出经过OPT 算法处理之后的结果。
2.4 输入新的页面引用串该模块的主要功能是用户自己输入新的页面引用串,系统默认的字符串是0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0,用户可以自定义全新的20个数字页面引用串。
页面置换常用的算法总结
页⾯置换常⽤的算法总结本⽂转载⾃:进程运⾏时,若其访问的页⾯不在内存⽽需将其调⼊,但内存已⽆空闲空间时,就需要从内存中调出⼀页程序或数据,送⼊磁盘的对换区。
选择调出页⾯的算法就称为页⾯置换算法。
好的页⾯置换算法应有较低的页⾯更换频率,也就是说,应将以后不会再访问或者以后较长时间内不会再访问的页⾯先调出。
常见的置换算法有以下四种。
1. 最佳置换算法(OPT)最佳(Optimal, OPT)置换算法所选择的被淘汰页⾯将是以后永不使⽤的,或者是在最长时间内不再被访问的页⾯,这样可以保证获得最低的缺页率。
但由于⼈们⽬前⽆法预知进程在内存下的若千页⾯中哪个是未来最长时间内不再被访问的,因⽽该算法⽆法实现。
最佳置换算法可以⽤来评价其他算法。
假定系统为某进程分配了三个物理块,并考虑有以下页⾯号引⽤串:7, 0, 1, 2, 0, 3, 0, 4, 2, 3, 0, 3, 2, 1, 2, 0, 1, 7, 0, 1进程运⾏时,先将7, 0, 1三个页⾯依次装⼊内存。
进程要访问页⾯2时,产⽣缺页中断,根据最佳置换算法,选择第18次访问才需调⼊的页⾯7予以淘汰。
然后,访问页⾯0时,因为已在内存中所以不必产⽣缺页中断。
访问页⾯3时⼜会根据最佳置换算法将页⾯1淘汰……依此类推,如图3-26所⽰。
从图中可以看出⾤⽤最佳置换算法时的情况。
可以看到,发⽣缺页中断的次数为9,页⾯置换的次数为6。
访问页⾯70120304230321201701物理块1777222227物理块200004000物理块31133311缺页否√√ √√√√√√√图3-26 利⽤最佳置换算法时的置换图2. 先进先出(FIFO)页⾯置换算法优先淘汰最早进⼊内存的页⾯,亦即在内存中驻留时间最久的页⾯。
该算法实现简单,只需把调⼊内存的页⾯根据先后次序链接成队列,设置⼀个指针总指向最早的页⾯。
但该算法与进程实际运⾏时的规律不适应,因为在进程中,有的页⾯经常被访问。
【精品】页面置换算法实验报告
【精品】页面置换算法实验报告一、实验目的了解操作系统中的页面置换算法,并实现FIFO、LRU和Clock算法。
二、实验原理页面置换算法是操作系统中用到的一种算法,其作用是在内存不够用时,选择牺牲已经在内存中的一些页,腾出更多的空间给新的内容。
本次实验主要实现了FIFO、LRU和Clock算法。
1、FIFO算法FIFO算法是最简单的页面置换算法,它采用先进先出的原则,即最先进入内存的页面应该最早被替换出去。
该算法的实现非常简单,只需要维护一个队列即可。
当需要置换页面时,选择队列的第一个页面进行替换即可。
2、LRU算法LRU算法是Least Recently Used的缩写,即最近最少使用算法。
该算法的核心思想是选择最久没有被使用的页面进行替换。
为了实现该算法,需要维护记录页面使用时间的链表、栈或队列等结构。
3、Clock算法Clock算法也叫做二次机会算法,是一种改良的FIFO算法。
它是基于FIFO算法的思想,并且每个页面都设置了一个使用位(use bit),用于记录该页面是否被使用过。
当需要置换一个页面时,检查该页面的使用位,如果该页面的使用位为1,则将该页面的使用位设置为0并移到队列的末尾,表示该页面有“二次机会”继续待在内存中;如果该页面的使用位为0,则选择该页面进行替换。
三、实验过程本次实验采用Python语言实现页面置换算法,并使用样例进行测试。
1、FIFO算法实现FIFO算法的实现非常简单,只需要用一个队列来维护已经在内存中的页面,当需要置换页面时,选择队列的第一个元素即可。
代码如下:```pythonfrom collections import dequeclass FIFO:def __init__(self, frame_num):self.frame_num = frame_numself.frames = deque(maxlen=frame_num)def access(self, page):if page in self.frames:return Falseif len(self.frames) >= self.frame_num:self.frames.popleft()self.frames.append(page)return True```2、LRU算法实现LRU算法的实现需要维护一个记录页面使用时间的链表或队列。
最佳页面置换算法
最佳页面置换算法(Optimal)(非常专业)评价一个算法的优劣,可通过在一个特定的存储访问序列(页面走向)上运行它,并计算缺页数量来实现。
1 先入先出法(FIFO)最简单的页面置换算法是先入先出(FIFO)法。
这种算法的实质是,总是选择在主存中停留时间最长(即最老)的一页置换,即先进入内存的页,先退出内存。
理由是:最早调入内存的页,其不再被使用的可能性比刚调入内存的可能性大。
建立一个FIFO队列,收容所有在内存中的页。
被置换页面总是在队列头上进行。
当一个页面被放入内存时,就把它插在队尾上。
这种算法只是在按线性顺序访问地址空间时才是理想的,否则效率不高。
因为那些常被访问的页,往往在主存中也停留得最久,结果它们因变“老”而不得不被置换出去。
FIFO的另一个缺点是,它有一种异常现象,即在增加存储块的情况下,反而使缺页中断率增加了。
当然,导致这种异常现象的页面走向实际上是很少见的。
现在来看下4块的情况:0 1 2 3 2 1 3 2 5 2 3 6 2 1 4 2【解答】刚开始内存并没有这个作业,所以发生缺页中断一次。
作业的0号页进入内存。
(1次缺页中断)而页1又不在内存,又发生缺页中断一次。
作业页1进入内存。
(2次缺页中断) 页2不在内存,发生缺页中断。
页2进入内存。
(3次缺页中断)页3不在内存,发生缺页中断。
页3进入内存。
(4次缺页中断)接下来调入页2,页1,页3,页2。
由于都在内存中,并不发生缺页中断。
页5不在内存,发生缺页中断。
页5进入内存,页5置换页0。
(5次缺页中断) 接下来调入页2,页3。
由于都在内存中,并不发生缺页中断。
页6不在内存,发生缺页中断。
页6进入内存。
页6置换页1。
(6次缺页中断) 页2在内存,不发生缺页中断。
页1不在内存(在发生第6次缺页中断时被置换了),发生缺页中断。
页1进入内存,页2被置换。
(7次缺页中断)页4置换页3,页4进入内存。
(8次缺页中断)现在调入页2,但页2在发生第7次缺页中断时被置换掉了。
常见页面置换算法图解
常见页⾯置换算法图解前⾔缓存⽂件置换的原因是电脑存储器空间固定,不可能将服务器上所有数据都加载在存储空间中,当需要调⽤不⽤的数据时,那么势必需要将需要的数据进来存储空间替换原有数据常见的缓存⽂件置换⽅法有:先进先出算法(FIFO):最先进⼊的内容作为替换对象最久未使⽤算法(LRU):最久没有访问的内容作为替换对象最近最少使⽤算法(LFU):最近最少使⽤的内容作为替换对象⾮最近使⽤算法(NMRU):在最近没有使⽤的内容中随机选择⼀个作为替换对象内存的平均引⽤时间为:其中T= 内存平均引⽤时间m= 未命中率 = 1 - (命中率)Tm= 未命中时访问主内存需要的时间 (或者在多层缓存中对下级缓存的访问时间)Th= 延迟,即命中时引⽤缓存的时间E= 各种次级因素, 如多处理器系统中的队列效应衡量缓存的指标主要有两个:延迟和命中率。
同时也存在其他⼀些次级因素影响缓存的性能。
缓存的命中率是指需要的对象在缓存中被找到的频率。
⾼效的置换策略会保留较多的实⽤信息来提升命中率(在缓存⼤⼩⼀定的情况下)。
缓存的延迟是指命中后,从发出请求到缓存返回指定对象所需的时间。
快速的置换策略通常会保留较少的置换信息,甚⾄不保留信息,来减少维护该信息所需要的时间。
每种置换策略都是在命中率和置换之间妥协。
先进先出算法(FIFO)如上图,在⼀个队列中,如果队列未满,添加资源时添加在末尾,如果队列资源已经满了,那么再添加资源时需要先将队列头部的资源移除,腾出空间后再将待添加的资源加⾄队列尾。
代码实现:public class FIFO implements Cacheable {private int maxLength = 0;private Queue<Object> mQueue = null;public FIFO(int _maxLength) {... ...}@Overridepublic void offer(Object object) {if (mQueue == null) {throw new NullPointerException("策略队列对象为空");}// check is need swap or notif (mQueue.size() == maxLength) {clean();}mQueue.offer(object);}@Overridepublic void visitting(Object object) {System.out.println("Visited " + object);}private void clean() {mQueue.poll();}}最久未使⽤算法(LRU)最久未使⽤算法图⽰:对⽐FIFO原理图和LRU原理图,可以很明显地看到只是在被使⽤的资源部分有⼀些⼩的改动。
O p t i m a l 最 佳 页 面 置 换 算 法
清华大学操作系统公开课(五)页面置换算法下面介绍在虚拟存储管理中有哪些页面置换算法。
局部页面置换算法最优页面置换算法(OPT,optimal)? 先进先出算法(FIFO)? 最近最久未使用算法(LRU,Least Recently Used)? 时钟页面置换算法(Clock)? 最不常用算法(LFU,Least Frequently Used)? Belady现象? LRU、FIFO和Clock的比较全局页面置换算法工作集模型? 工作集页置换算法? 缺页率置换算法2.页面置换算法当缺页中断发生,需要调入新的页面而内存已满时,选择内存当中哪个物理页面被置换。
尽可能地减少页面的换进换出次数(即缺页中断的次数)。
具体来说,把未来不再使用的或短期内较少使用的页面换出,通常只能在局部性原理指导下依据过去的统计数据来进行预测。
页面锁定(frame locking)页面锁定技术是用来锁定物理内存中不应该被换出的内存数据。
用于描述必须常驻内存的操作系统的关键部分或时间关键(time-critical)的应用进程。
实现的方法是:在页表中添加锁定标志位(lock bit)。
我们该如何评估不同页面置换算法的优劣?通过程序运行时的效率来比较是不容易实现的,可以记录下一个进程对页面的访问轨迹,然后模拟一个页面置换的行为并且记录产生页缺失的数量,以此比较优劣。
3.最优页面置换算法基本思路当一个缺页中断发生时,对于保存在内存当中的每一个逻辑页面,计算在它下一次访问之前,还需要等待多长时间,从中选择等待时间最长的那个,作为被置换的页面。
这只是一种理想情况,在实际系统中是无法实现的,因为操作系统无从知道每一个页面要等待多长时间后才会再次被访问。
可用作其它算法的性能评价的依据(在一个模拟器上运行某个城西,并记录每一次的页面访问情况,在第二遍运行时即可使用最优算法)。
物理内存大小为4个帧,刚开始存入的物理帧是a、b、c、d,一共需要载入5个帧,当访问e时发生缺页中断,此时置换入e,置换出d,因为下一次访问d的等待时间最长,暂时用不到d,可以让d待在外存。
页面置换算法
页⾯置换算法 页⾯置换算法根据置换页⾯的选择范围分为局部页⾯置换算法和全局页⾯置换算法。
局部页⾯置换算法只置换本进程内的物理页⾯,进程中⼀个页⾯进内存,就代表⼀个页⾯已经被替换出内存,所以⼀个进程所占⽤的物理页⾯的总数是确定的。
全局页⾯置换算法置换内存中所有可换出的物理页⾯,即换进内存的是进程A的页⾯,换出内存的可能是进程B的页⾯,所以进程在内存中占⽤的页⾯总数是可变的。
⼀、局部页⾯置换算法: 1.最优页⾯置换算法: 预测未来最晚被使⽤的页⾯,然后把它置换到外存中去。
缺页最少,但⽆法实现,只是⼀种理想状态,所以作为其他置换算法的评测标准。
2.先进先出算法 有⼀个逻辑页⾯的链表,链表头在内存中呆的时间最长,链表尾最短,所以每次置换都是把链表头的页⾯置换到外存中去,把新加进来的页⾯放到链表尾。
这个算法虽然简单,但是造成的缺页率⽐较⾼,性能⽐较差。
3.最近最久未使⽤算法/最近最少使⽤算法 替换内存中上次被引⽤时间最早的⼀个页⾯。
可以⽤链表或栈来记录按被引⽤时间长短排序的页⾯,但不管是栈还是链表,每次替换都会牵扯到整个链表或栈的移动,算法性能⽐较好,但开销⽐较⼤。
4.时钟算法 ⽤环形链表存储各页⾯。
初始化时各页⾯的访问位为0。
如果不缺页,则把相应页⾯的访问位设置为1。
如果缺页,则从最先进⼊链表的页⾯开始遍历,遇到访问位为1的页⾯,则访问位设置为0;遇到访问位为0的页⾯,则把它替换到外存中去,然后把需要的页⾯替换进内存,且访问位为1。
然后把当前指针移到下⼀个页⾯上,再进⾏下⼀次页⾯请求处理。
⽐FIFO多了个访问位来记录页⾯的访问情况,但⽐LRU少了访问的先后顺序,所以说是FIFO和LRU的折中。
5.改进的时钟算法 在页⾯中增加了修改位,1为修改过,0为未修改过。
因为当发⽣缺页中断时,把未修改过的页⾯替换进外存就相当于把页⾯直接从内存中删除,因为内存和外存中所对应的该页⾯的内容相同,处理时间只有⼀次缺页中断访问外存的时间。
页面置换算法实验报告
页面置换算法实验报告一、实验目的本次实验的目的是通过模拟页面置换算法的过程,了解不同算法的优缺点,掌握算法的实现方法,以及对算法的性能进行评估。
二、实验原理页面置换算法是操作系统中的一个重要概念,它是为了解决内存不足的问题而产生的。
当系统中的进程需要使用内存时,如果内存已经被占满,就需要将一些页面从内存中置换出去,以便为新的页面腾出空间。
页面置换算法就是用来决定哪些页面应该被置换出去的算法。
常见的页面置换算法有以下几种:1. 最佳置换算法(OPT)最佳置换算法是一种理论上的最优算法,它总是选择最长时间内不会被访问的页面进行置换。
但是,由于无法预测未来的页面访问情况,因此最佳置换算法无法在实际中使用。
2. 先进先出置换算法(FIFO)先进先出置换算法是一种简单的置换算法,它总是选择最先进入内存的页面进行置换。
但是,这种算法容易出现“抖动”现象,即频繁地将页面置换出去,然后再将其置换回来。
3. 最近最久未使用置换算法(LRU)最近最久未使用置换算法是一种比较常用的置换算法,它总是选择最长时间未被访问的页面进行置换。
这种算法可以避免“抖动”现象,但是实现起来比较复杂。
4. 时钟置换算法(Clock)时钟置换算法是一种改进的FIFO算法,它通过维护一个环形链表来实现页面置换。
当需要置换页面时,算法会从当前位置开始扫描链表,如果找到一个未被访问的页面,则将其置换出去。
如果扫描一圈后都没有找到未被访问的页面,则将当前位置的页面置换出去。
三、实验过程本次实验使用Python语言编写了一个页面置换算法模拟程序,可以模拟上述四种算法的过程,并输出算法的性能指标。
程序的主要流程如下:1. 读取输入文件,获取页面访问序列和内存大小等参数。
2. 根据选择的算法,初始化相应的数据结构。
3. 遍历页面访问序列,模拟页面置换的过程。
4. 输出算法的性能指标,包括缺页率、页面置换次数等。
下面分别介绍四种算法的实现方法。
1. 最佳置换算法(OPT)最佳置换算法需要预测未来的页面访问情况,因此需要遍历整个页面访问序列,找到最长时间内不会被访问的页面。
操作系统-页面置换算法课件
第一章 操作系统引论
图5-3 利用最佳页面置换算法时的置换图
3
第一章 操作系统引论
2. 先进先出(FIFO)页面置换算法 FIFO算法是最早出现的置换算法。该算法总是淘汰最先 进入内存的页面,即选择在内存中驻留时间最久的页面予以 淘汰。该算法实现简单,只需把一个进程已调入内存的页面 按先后次序链接成一个队列,并设置一个指针,称为替换指 针,使它总是指向最老的页面。但该算法与进程实际运行的 规律不相适应,因为在进程中,有些页面经常被访问,比如, 含有全局变量、常用函数、例程等的页面,FIFO算法并不能 保证这些页面不被淘汰。
9
第一章 操作系统引论
图5-6 某进程具有8个页面时的LRU访问情况
10
第一章 操作系统引论
2) 栈 可利用一个特殊的栈保存当前使用的各个页面的页面号。 每当进程访问某页面时,便将该页面的页面号从栈中移出, 将它压入栈顶。因此,栈顶始终是最新被访问页面的编号, 而栈底则是最近最久未使用页面的页面号。假定现有一进程, 它分有五个物理块,所访问的页面的页面号序列为:
4
第一章 操作系统引论
图5-4 利用FIFO置换算法时的置换图
5
第一章 操作系统引论
5.3.2 最近最久未使用和最少使用置换算法 1. LRU(Least Recently Used)置换算法的描述 FIFO置换算法的性能之所以较差,是因为它所依据的条
件是各个页面调入内存的时间,而页面调入的先后并不能反 映页面的使用情况。最近最久未使用(LRU)的页面置换算法 是根据页面调入内存后的使用情况做出决策的。
1
第一章 操作系统引论
5.3.1 最佳置换算法和先进先出置换算法 1. 最佳(Optimal)置换算法 最佳置换算法是由Belady于1966年提出的一种理论上的
fifo 页面置换算法
fifo 页面置换算法页面置换算法是操作系统中一种重要的内存管理技术,用于决定当内存中某个时间点所包含的页面(即帧)数量大于物理内存容量时,应该淘汰哪个页面,以便为新的页面腾出空间。
FIFO (FirstInFirstOut,先进先出)页面置换算法是一种常见的算法,其基本思想是优先淘汰最先进入内存的页面。
一、算法原理FIFO页面置换算法的基本原理是,当需要为新的页面分配内存时,选择最早进入内存的页面进行淘汰。
这种算法的优点是实现简单,缺点是对频繁调用的页面影响较大,因为这些页面最先进入内存,所以被淘汰的可能性也较大。
但是它能够确保被淘汰的页面是最早进入内存的页面,因此它能够提供一定的公平性。
二、算法步骤FIFO页面置换算法的实施步骤如下:1.记录每个页面进入和离开内存的时间;2.当需要为新的页面分配内存时,比较该页面与其最先进入内存的时间;3.优先淘汰最先进入内存的页面;4.将新页面放入空出的帧中。
三、算法优缺点1.优点:a.实现简单,易于实现;b.在许多场景下能提供较好的性能;c.有利于保持页面的有序性。
2.缺点:a.频繁调用的页面被淘汰的可能性较大,可能导致频繁的页面加载和卸载操作;b.对于某些应用场景可能不够高效,因为一些页面可能长时间在内存中占据空间,而不会被频繁使用。
因此需要对其进行优化,以便在减少页面的浪费和提高系统性能之间找到平衡。
四、应用场景FIFO页面置换算法适用于各种操作系统和应用程序,包括但不限于Web服务器、数据库系统、桌面环境等。
它尤其适用于那些对响应速度要求较高且对内存使用效率要求不高的场景。
例如,一些网页浏览、邮件阅读等应用场景,由于页面加载频率较高,FIFO算法可能会影响性能。
五、总结总的来说,FIFO页面置换算法是一种简单易行的内存管理技术,但在实际应用中需要根据具体场景进行优化。
在实际操作中,需要根据应用的特点和需求选择合适的页面置换算法,以提高系统的性能和稳定性。
操作系统——模拟页面置换算法(FIFO——先入先出、LRU——最近最少使用、LFU——最近。。。
操作系统——模拟页⾯置换算法(FIFO——先⼊先出、LRU——最近最少使⽤、LFU——最近。
操作系统——模拟页⾯置换算法(FIFO——先⼊先出、LRU——最近最少使⽤、LFU——最近最不常使⽤),计算置换率(包含程序框图)导语:1. FIFO页⾯置换算法:最简单的页⾯置换算法。
这种算法的基本思想是:当需要淘汰⼀个页⾯时,总是选择驻留主存时间最长的页⾯进⾏淘汰,即先进⼊主存的页⾯先淘汰。
(看时间)2. LRU页⾯置换算法:最近最少使⽤,简单来说就是将数据块中,每次使⽤过的数据放在数据块的最前端,然后将存在的时间最长的,也就是数据块的末端的数据置换掉。
(看时间)3. LFU页⾯置换算法:近期最少使⽤算法,选择近期最少访问的页⾯作为被替换的页⾯,如果⼀个数据在最近⼀段时间内使⽤次数很少,那么在将来⼀段时间内被使⽤的可能性也很⼩。
(看次数)4. 置换率与与缺页率不同。
置换率⽤置换次数算,缺页率⽤缺页中断次数算。
FIFO页⾯置换算法:Linux效果图(采⽤UOS + VScode + g++)程序框图C++代码(FIFO):#include<iostream>using namespace std;static int mnum;//物理块数static int pnum;//页⾯⾛向static int count=0;//页⾯置换次数static int *analogblock;//模拟物理块static int *block;//物理块static int *process;//随机页⾯访问序列int judge(int a[],int n,int x) //判断数组中是否已有x,若有返回其下标值,没有则返回-1 {int i;for (i=0;i<n;i++)if(x==a[i])return i;return -1;}void replace(int y,int mnum,int x)//⽤于物理块页⾯置换,y是⽤来置换的页⾯,x是被置换的页⾯ {int i;for (i=0;i<mnum;i++)if(x==block[i])block[i]=y;}int main() {int i;int maxanalogblock=-1;//模仿队列的定义int x;cout<<"请输⼊页框⼤⼩物理块数:\n";cin>>mnum;if(mnum>999999) {cout<<"输⼊超出控制⼤⼩!"<<endl;return 0;}cout<<"⾃动⽣成的内存块需求序列个数:\n";cin>>pnum;if(pnum>999999) {cout<<"输⼊超出控制⼤⼩!"<<endl;return 0;}analogblock=new int[mnum];block=new int[mnum];process=new int[pnum];for (i=0;i<mnum;i++) analogblock[i]=-1;for (i=0;i<mnum;i++) block[i]=-1;///////////////////////随机产⽣页⾯⾛向序列cout<<"产⽣随机序列如下:\n";srand( (unsigned)time( NULL ) );//以time函数值(即当前时间)作为种⼦数,保证两次产⽣序列的随机性for (i=0; i<pnum; i++) {process[i] = rand()%10;cout<<process[i]<<" ";}cout<<endl;//////////////////////cout<<"先进先出(FIFO)页⾯置换算法,结果: \n\n";//////////////////////for (x=0;x<pnum;x++) //⾃动读数 {//读⼀个序列号,输出当前数组元素cout<<"真实物理块情况:";for (i=0;i<mnum;i++) {if(block[i]!=-1)cout<<block[i]<<" ";}cout<<"模拟物理块情况:";for (i=0;i<mnum;i++) {if(analogblock[i]!=-1)cout<<analogblock[i]<<" ";}//////////////////////////maxanalogblock++;//读数后maxanalogblock⾃动+1if(maxanalogblock<mnum) //若在物理块范围内 {if(judge(analogblock,mnum,process[x])==-1)//若数组中不存在待插⼊元素 {analogblock[maxanalogblock]=process[x];//新元素从尾部插⼊block[maxanalogblock]=process[x];//新元素从尾部插⼊cout<<" 第"<<x+1<<"次访问,页⾯"<<process[x]<<" 缺页中断调⼊页⾯"<<process[x]<<endl;} else //若数组中存在待插⼊元素 {maxanalogblock--;//因为没有插⼊新元素,回滚maxanalogblock值cout<<" 第"<<x+1<<"次访问,页⾯"<<process[x]<<" 已存在直接访问"<<endl;}} else //超过物理块数的元素 {if(judge(analogblock,mnum,process[x])==-1)//若数组中不存在待插⼊元素 {//队列法插⼊(尾部元素出,新元素从头部⼊)cout<<" 第"<<x+1<<"次访问,页⾯"<<process[x]<<" 缺页中断页⾯"<<process[x]<<"置换出页⾯"<<analogblock[0]<<endl; replace(process[x],mnum,analogblock[0]);//置换物理块中页⾯for (i=0;i<mnum-1;i++)LRU 页⾯置换算法:Linux 效果图(采⽤UOS + VScode + g++)程序框图C++代码(LRU): analogblock[i]=analogblock[i+1];analogblock[mnum-1]=process[x];//////////////////maxanalogblock--;//因为没有插⼊新元素,回滚maxanalogblock 值count++;} else //若数组中存在待插⼊元素 {maxanalogblock--;//因为没有插⼊新元素,回滚maxanalogblock 值cout<<" 第"<<x+1<<"次访问,页⾯"<<process[x]<<" 已存在 直接访问"<<endl;}}}//读⼀个序列号,输出当前数组元素cout<<"真实物理块情况:";for (i=0;i<mnum;i++) {if(block[i]!=-1) cout<<block[i]<<" ";}cout<<"模拟物理块情况:";for (i=0;i<mnum;i++) {if(analogblock[i]!=-1)cout<<analogblock[i]<<" ";}cout<<endl<<"页⾯换算次数为:"<<count<<endl;cout<<"置换率为:"<<(float)count/pnum<<endl;return 0;}//g++ test71.cpp -o test71 -lpthread&&./test71#include<iostream>using namespace std;static int mnum;//物理块数static int pnum;//页⾯⾛向static int count=0;//页⾯置换次数static int *analogblock;//模拟物理块static int *block;//物理块static int *process;//随机页⾯访问序列int judge(int a[],int n,int x) //判断数组中是否已有x ,若有返回其下标值,没有则返回-1 {int i;for (i=0;i<n;i++)if(x==a[i])return i;return -1;}void replace(int y,int mnum,int x)//⽤于物理块页⾯置换,y是⽤来置换的页⾯,x是被置换的页⾯ { int i;for (i=0;i<mnum;i++)if(x==block[i])block[i]=y;}void move(int a[],int n,int i) //移动下标为i的元素到尾部 {int j;int m=a[i];for (j=i;j<n-1;j++)a[j]=a[j+1];a[n-1]=m;}int main() {int i;int maxanalogblock=-1;//模仿栈的定义int x;cout<<"请输⼊页框⼤⼩物理块数:\n";cin>>mnum;if(mnum>999999) {cout<<"输⼊超出控制⼤⼩!"<<endl;return 0;}cout<<"⾃动⽣成的内存块需求序列个数:\n";cin>>pnum;if(pnum>999999) {cout<<"输⼊超出控制⼤⼩!"<<endl;return 0;}analogblock=new int[mnum];block=new int[mnum];process=new int[pnum];for (i=0;i<mnum;i++) analogblock[i]=-1;///////////////////////随机产⽣页⾯⾛向序列cout<<"产⽣随机序列如下:\n";srand( (unsigned)time( NULL ) );//以time函数值(即当前时间)作为种⼦数,保证两次产⽣序列的随机性for (i=0; i<pnum; i++) {process[i] = rand()%10;cout<<process[i]<<" ";}cout<<endl;//////////////////////cout<<"最近最少使⽤(LRU)页⾯置换算法,结果: \n\n";//////////////////////for (x=0;x<pnum;x++) //⾃动读数 {//读⼀个序列号,输出当前数组元素cout<<"真实物理块情况:";for (i=0;i<mnum;i++) {if(block[i]!=-1)cout<<block[i]<<" ";}cout<<"模拟物理块情况:";for (i=0;i<mnum;i++) {if(analogblock[i]!=-1)cout<<analogblock[i]<<" ";}//////////////////////////maxanalogblock++;//读数后maxanalogblock⾃动+1LFU 页⾯置换算法:Linux 效果图(采⽤UOS + VScode + g++)程序框图 if(maxanalogblock<mnum) //若在物理块范围内 {if(judge(analogblock,mnum,process[x])==-1)//若数组中不存在待插⼊元素 {analogblock[maxanalogblock]=process[x];//新元素从尾部插⼊block[maxanalogblock]=process[x];//新元素从尾部插⼊cout<<" 第"<<x+1<<"次访问,页⾯"<<process[x]<<" 缺页中断 调⼊页⾯"<<process[x]<<endl;} else //若数组中存在待插⼊元素 {move(analogblock,maxanalogblock,judge(analogblock,mnum,process[x]));//移动下标为i 的元素到尾部maxanalogblock--;//因为没有插⼊新元素,回滚maxanalogblock 值cout<<" 第"<<x+1<<"次访问,页⾯"<<process[x]<<" 已存在 直接访问"<<endl;}} else //超过物理块数的元素 {if(judge(analogblock,mnum,process[x])==-1)//若数组中不存在待插⼊元素 {//栈法插⼊(第⼀个元素出,后⾯元素前移,新元素从尾部⼊)cout<<" 第"<<x+1<<"次访问,页⾯"<<process[x]<<" 缺页中断 页⾯"<<process[x]<<"置换出页⾯"<<analogblock[0]<<endl; replace(process[x],mnum,analogblock[0]);//物理块中页⾯置换for (i=0;i<mnum-1;i++)analogblock[i]=analogblock[i+1];analogblock[mnum-1]=process[x];//////////////////maxanalogblock--;//因为没有插⼊新元素,回滚maxanalogblock 值count++;} else //若数组中存在待插⼊元素 {move(analogblock,mnum,judge(analogblock,mnum,process[x]));//移动下标为i 的元素到尾部maxanalogblock--;//因为没有插⼊新元素,回滚maxanalogblock 值cout<<" 第"<<x+1<<"次访问,页⾯"<<process[x]<<" 已存在 直接访问"<<endl;}}}//读⼀个序列号,输出当前数组元素cout<<"真实物理块情况:";for (i=0;i<mnum;i++) {if(block[i]!=-1)cout<<block[i]<<" ";}cout<<"模拟物理块情况:";for (i=0;i<mnum;i++) {if(analogblock[i]!=-1)cout<<analogblock[i]<<" ";}cout<<endl<<"页⾯换算次数为:"<<count<<endl;cout<<"置换率为:"<<(float)count/pnum<<endl;return 0;}//g++ test72.cpp -o test72 -lpthread&&./test72C++代码(LFU):#include<iostream>using namespace std;static int mnum;//物理块数static int pnum;//页⾯⾛向static int count=0;//页⾯置换次数static int **analogblock;//模拟物理块static int *block;//物理块static int *process;//随机页⾯访问序列int judge(int *a[],int n,int x) //判断数组中是否已有x,若有返回其下标值,没有则返回-1 {int i;for (i=0;i<n;i++)if(x==a[i][0])return i;return -1;}void replace(int y,int mnum,int x)//⽤于物理块页⾯置换,y是⽤来置换的页⾯,x是被置换的页⾯ { int i;for (i=0;i<mnum;i++)if(x==block[i])block[i]=y;}void move(int *a[],int n,int i) //移动下标为i的元素,⽐较访问次数次多少进⾏前进 {int j;int m=a[i][0];int m2=a[i][1];for (j=i;j<n-1;j++) {if(m2>=a[j+1][1]) {a[j][0]=a[j+1][0];a[j][1]=a[j+1][1];a[j+1][0]=m;a[j+1][1]=m2;}}}int main() {int i;int maxanalogblock=-1;//模仿栈的定义int x;//动态数组初始化cout<<"请输⼊页框⼤⼩物理块数:\n";cin>>mnum;if(mnum>999999) {cout<<"输⼊超出控制⼤⼩!"<<endl;return 0;}cout<<"⾃动⽣成的内存块需求序列个数:\n";cin>>pnum;if(pnum>999999) {cout<<"输⼊超出控制⼤⼩!"<<endl;return 0;}analogblock=(int**) (new int[mnum]);block=new int[mnum];process=new int[pnum];for (i=0;i<mnum;i++) analogblock[i]=(int*) new int[2];//⽤于保存页⾯号和访问次数for (i = 0; i < mnum; i++) {analogblock[i][0]=-1;analogblock[i][1]=0;}///////////////////////随机产⽣页⾯⾛向序列cout<<"产⽣随机序列如下:\n";srand( (unsigned)time( NULL ) );//以time函数值(即当前时间)作为种⼦数,保证两次产⽣序列的随机性for (i=0; i<pnum; i++) {process[i] = rand()%10;cout<<process[i]<<" ";}cout<<endl;//////////////////////cout<<"最近最不常使⽤(LFU)页⾯置换算法,结果: \n\n";//////////////////////for (x=0;x<pnum;x++) //⾃动读数 {//读⼀个序列号,输出当前数组元素cout<<"真实物理块情况:";for (i=0;i<mnum;i++) {if(block[i]!=-1)cout<<block[i]<<" ";}cout<<"模拟物理块情况:";for (i=0;i<mnum;i++) {if(analogblock[i][0]!=-1)cout<<analogblock[i][0]<<" ";//<<"访问次数"<<analogblock[i][1]<<" "}//////////////////////////maxanalogblock++;//读数后maxanalogblock⾃动+1if(maxanalogblock<mnum) //若在物理块范围内 {if(judge(analogblock,mnum,process[x])==-1)//若数组中不存在待插⼊元素 {analogblock[0][0]=process[x];//新元素从头部插⼊analogblock[0][1]=1;block[maxanalogblock]=process[x];//新元素从尾部插⼊move(analogblock,mnum,0);//移动下标为i的元素到相同访问次数页⾯的顶部cout<<" 第"<<x+1<<"次访问,页⾯"<<process[x]<<" 缺页中断调⼊页⾯"<<process[x]<<endl;} else //若数组中存在待插⼊元素 {// move(analogblock,maxanalogblock,judge(analogblock,mnum,process[x]));//移动下标为i的元素到尾部analogblock[judge(analogblock,mnum,process[x])][1]++;move(analogblock,mnum,judge(analogblock,mnum,process[x]));//移动下标为i的元素到相同访问次数页⾯的顶部maxanalogblock--;//因为没有插⼊新元素,回滚maxanalogblock值cout<<" 第"<<x+1<<"次访问,页⾯"<<process[x]<<" 已存在直接访问"<<endl;}} else //超过物理块数的元素 {if(judge(analogblock,mnum,process[x])==-1)//若数组中不存在待插⼊元素 {//栈法插⼊(新元素从头部⼊,替换掉头部)cout<<" 第"<<x+1<<"次访问,页⾯"<<process[x]<<" 缺页中断页⾯"<<process[x]<<"置换出页⾯"<<analogblock[0][0]<<endl; replace(process[x],mnum,analogblock[0][0]);//物理块中页⾯置换analogblock[0][0]=process[x];analogblock[0][1]=1;move(analogblock,mnum,0);//移动下标为i的元素相同访问次数页⾯的顶部//////////////////maxanalogblock--;//因为没有插⼊新元素,回滚maxanalogblock值count++;} else //若数组中存在待插⼊元素 {analogblock[judge(analogblock,mnum,process[x])][1]++;move(analogblock,mnum,judge(analogblock,mnum,process[x]));//移动下标为i的元素到相同访问次数页⾯的顶部maxanalogblock--;//因为没有插⼊新元素,回滚maxanalogblock值cout<<" 第"<<x+1<<"次访问,页⾯"<<process[x]<<" 已存在直接访问"<<endl; }}}//读⼀个序列号,输出当前数组元素cout<<"真实物理块情况:";for (i=0;i<mnum;i++) {if(block[i]!=-1)cout<<block[i]<<" ";}cout<<"模拟物理块情况:";for (i=0;i<mnum;i++) {if(analogblock[i][0]!=-1)cout<<analogblock[i][0]<<" ";}cout<<endl<<"页⾯换算次数为:"<<count<<endl;cout<<"置换率为:"<<(float)count/pnum<<endl;return 0;}//g++ test73.cpp -o test73 -lpthread&&./test73。
最佳页面置换算法
最佳页面置换算法(Optimal)(非常专业)评价一个算法的优劣,可通过在一个特定的存储访问序列(页面走向)上运行它,并计算缺页数量来实现。
1 先入先出法(FIFO)最简单的页面置换算法是先入先出(FIFO)法。
这种算法的实质是,总是选择在主存中停留时间最长(即最老)的一页置换,即先进入内存的页,先退出内存。
理由是:最早调入内存的页,其不再被使用的可能性比刚调入内存的可能性大。
建立一个FIFO队列,收容所有在内存中的页。
被置换页面总是在队列头上进行。
当一个页面被放入内存时,就把它插在队尾上。
这种算法只是在按线性顺序访问地址空间时才是理想的,否则效率不高。
因为那些常被访问的页,往往在主存中也停留得最久,结果它们因变“老”而不得不被置换出去。
FIFO的另一个缺点是,它有一种异常现象,即在增加存储块的情况下,反而使缺页中断率增加了。
当然,导致这种异常现象的页面走向实际上是很少见的。
现在来看下4块的情况:0 1 2 3 2 1 3 2 5 2 3 6 2 1 4 2【解答】刚开始内存并没有这个作业,所以发生缺页中断一次。
作业的0号页进入内存。
(1次缺页中断)而页1又不在内存,又发生缺页中断一次。
作业页1进入内存。
(2次缺页中断) 页2不在内存,发生缺页中断。
页2进入内存。
(3次缺页中断)页3不在内存,发生缺页中断。
页3进入内存。
(4次缺页中断)接下来调入页2,页1,页3,页2。
由于都在内存中,并不发生缺页中断。
页5不在内存,发生缺页中断。
页5进入内存,页5置换页0。
(5次缺页中断) 接下来调入页2,页3。
由于都在内存中,并不发生缺页中断。
页6不在内存,发生缺页中断。
页6进入内存。
页6置换页1。
(6次缺页中断) 页2在内存,不发生缺页中断。
页1不在内存(在发生第6次缺页中断时被置换了),发生缺页中断。
页1进入内存,页2被置换。
(7次缺页中断)页4置换页3,页4进入内存。
(8次缺页中断)现在调入页2,但页2在发生第7次缺页中断时被置换掉了。
【操作系统】页面置换算法(最佳置换算法)(C语言实现)
【操作系统】页⾯置换算法(最佳置换算法)(C语⾔实现)【操作系统】页⾯置换算法(最佳置换算法)(C语⾔实现)(编码⽔平较菜,写博客也只是为了个⼈知识的总结和督促⾃⼰学习,如果有错误,希望可以指出)1.页⾯置换算法:在地址映射过程中,若在页⾯中发现所要访问的页⾯不在内存中,则产⽣缺页中断。
当发⽣缺页中断时,如果操作系统内存中没有空闲页⾯,则操作系统必须在内存选择⼀个页⾯将其移出内存,以便为即将调⼊的页⾯让出空间。
⽽⽤来选择淘汰哪⼀页的规则叫做页⾯置换算法。
⼀个好的页⾯置换算法,应具有较低的页⾯更换频率。
从理论上讲,应该保留最近重复访问的页⾯,将以后都不再访问或者很长时间内不再访问的页⾯调出。
----百度百科2.具体的页⾯置换算法:2.1 最佳置换算法:⼀个进程在内存的若⼲个页⾯中,哪⼀个页⾯是未来最长时间内不再被访问的,那么如果发⽣缺页中断时,就将该页⾯换出,以便存放后⾯调⼊内存中的页⾯。
1.这是计算机操作系统(第四版)中的⼀个例⼦。
系统⾸先为进程分配了三个物理块。
上⾯⼀排数字是作业号。
在转满三个物理块后,要访问2号作业,2号作业不在内存,所以会发⽣缺页中断,然后系统需要将2号作业调⼊内存,但是此时物理块已经装满。
2.依据最佳置换算法,会将7号页换出(0号页在2号页后第1个就会被访问,1号页在2号页后第10个会被访问,7号页在2号页后第14个会被访问,7号页在已经装⼊内存的作业中是未来最长时间不会被访问的,所以换出7号页)。
3.后⾯依次类推。
2.2 先进先出算法:如果发⽣缺页中断,需要换出⼀个页⾯的时候,总是选择最早进⼊内存的页⾯,即选择在内存中驻留时间最久的页⾯进⾏换出。
有点不清楚。
就是每次发⽣缺页就将最早进⼊内存的页⾯换出,然后将刚调⼊的页⾯换⼊该物理块。
2.3 最近最久未使⽤(LRU)置换算法:LRU算法是缺页中断发⽣时选择最久未使⽤的页⾯进⾏换出。
这个算法其实也很好判断。
分享⼀个⼩技巧。
内存分配了k个物理块,发⽣缺页中断将要往内存调⼊某个页⾯的时候,在该页⾯往前⾯数K个物理块最前⾯的那个就会是要换出的,因为该页⾯最长时间未被使⽤过。
页面置换算法实验报告实验心得
页面置换算法实验报告实验心得
一、实验背景
页面置换算法是操作系统中的一个重要概念,是内存管理机制的核心内容之一。
页面置换算法是为了解决内存有限的问题,通过将不常用的页面换出,腾出空间给新的页面使用。
页面置换算法有多种,如LRU、FIFO、Clock等。
本次实验主要研究这几种置换算法的实现原理并通过编程实现。
二、实验过程
本次实验首先通过查阅相关资料了解了LRU、FIFO、Clock等页面置换算法的实现原理。
在了解了原理之后,我们通过使用Python语言编写程序来模拟这些算法的实现。
实验中,我们首先编写了一段程序来生成一个不同大小的访问页面序列,并将生成的页面序列分别输入到LRU、FIFO、Clock算法的模拟程序中,观察算法的置换效果,并通过比较不同算法的置换效果来评估各个算法的优劣。
三、实验结果
通过实验,我们观察到在不同的访问页面序列下,各个算法的置换效果都有所不同。
在某些情况下,FIFO算法的置换效果最好,而在
其他情况下,LRU算法的置换效果最佳。
同时,在一些场景下,Clock算法的置换效果也不错。
这说明不同的页面置换算法对于不同的内存使用情况有着不同的适用性。
四、实验心得
通过本次实验,我们深入了解了页面置换算法的实现原理,并通过编程实现了这些算法。
通过实验,我们发现在不同的场景下,不同的页面置换算法的效果会有所不同。
因此,在实际应用中,我们需要考虑内存使用情况,选择最适合的页面置换算法来提高内存利用率和程序性能。
同时,通过这次实验,我们也了解了Python编程的基本语法和常用库的使用方法,对我们的编程能力有了一定的提高。
页面置换算法实验总结
页面置换算法实验总结
在操作系统中,页面置换算法是为了解决内存不足的问题,当内存中的页面不足时,需要选择一些页面进行置换,将其换出到磁盘上,从而为新的页面腾出空间。
在本次实验中,我实现了三种页面置换算法,分别是FIFO(先进先出)、LRU(最近最少使用)和OPT(最佳置换)。
下面是对这三种算法的总结:
1. FIFO算法:FIFO算法是最简单的页面置换算法,它按照页面进入内存的顺序进行置换。
实验结果显示,FIFO算法在某些情况下可能会导致“抖动”现象,即不断发生页面置换,性能较差。
2. LRU算法:LRU算法是根据页面的使用历史进行置换,将最长时间没有被使用的页面置换出去。
实验结果显示,LRU算法相比于FIFO算法在减少页面抖动方面表现更好,但是实现起来较为复杂,需要维护一个访问历史记录的数据结构。
3. OPT算法:OPT算法是一种理想情况下的页面置换算法,它通过预测未来的页面访问情况来选择最佳的页面进行置换。
实验结果显示,OPT算法在减少页面抖动方面表现最好,但是实现起来较为困难,需要对未来的页面访问情况进行预测。
综上所述,不同的页面置换算法在不同的场景下有着不同的表现。
FIFO算法简单易实现,但性能较差;LRU算法在某些情况下能够较好地减少页面抖动;OPT算法在理论上是最佳的页面置换算法,但实现起来较为困难。
实际中的选择需要根据具体的应用场景
和系统需求来确定。
操作系统之页面置换算法(最佳置换OPT,先进先出FIFO,最近最久未使用LRU)
操作系统之页⾯置换算法(最佳置换OPT,先进先出FIFO,最近最久未使⽤LRU)最近学习操作系统时,实验要求实现常见的三种页⾯置换算法,博主按照书上要求试着编写,实现了案例,并记录在博客随记中,以便后续⾃⼰复习并也给需要的同学分享参考⼀下!⽔平有限,若有错,请悄悄告诉博主!博主好⽴即改正。
最佳置换算法(optimal replacement,OPT)是从内存中选择今后不再访问的页⾯或者在最长⼀段时间后才需要访问的页⾯进⾏淘汰。
如下例⼦:根据页⾯⾛向依次处理,得到最终的置换结果如下图表,整个页⾯缺页次数为7,缺页率为7/12=58%。
1 #include <iostream>2 #include <stdio.h>3 #include <stdlib.h>4#define N 125#define B 36using namespace std;78int pageArr[N]={1,2,3,4,1,2,5,1,2,3,4,5};//页⾯⾛向9int block[B]={0};//物理块3个,其数值是页号10 typedef struct FLAG {11int flags[B];12int counts;13 } FLAG;1415void opt(int pageArr[],int block[]);16int inBlock(int which);17int findFar(int next);18void Replace(int index,int value);19void disPlay();2021int main(void){22 cout << "begin:" <<endl;23 opt(pageArr,block);24 cout << "end!" <<endl;25return0;26 }2728void opt(int pageArr[],int block[]){29int getIndex;30for(int i=0;i<N;i++){31if(i<3){//前3页号#短缺#进队列32 block[i]=pageArr[i];33 printf("缺页:(null)-->%d\n",pageArr[i]);34 }35else {36if(i==3){37 disPlay();3839 }40if(inBlock(pageArr[i])!=-1){//下⼀个页⾯if在物理块中返回index并跳过,反-141 disPlay();4243continue;44 }45 getIndex=findFar(i+1);//从下⼀个页号,找到最远出现的页⾯,替换的下标46if(getIndex==-1){47 cout<<"error,not replace obj!"<<'\t';48 }49else{50 Replace(getIndex,pageArr[i]);//由下标找到上⼀组替换⽬标,⽤第⼆参数替换51 disPlay();5253 }54 }55 }56return;57 }5859//替换block中的物理块60void Replace(int index,int value){61 printf("缺页:%d--被替换为-->%d\n",block[index],value);62 block[index]=value;63return;64 }656667//找到最远出现的页⾯68int findFar(int next){69int index=-1;//error,默认返回不存在的索引70 FLAG myflag;71 myflag.flags[0]=0;72 myflag.flags[1]=0;73 myflag.flags[2]=0;74 myflag.counts=0;75int stop = N-next;76while(stop--){77 index=inBlock(pageArr[next++]);78if(index!=-1){79 myflag.flags[index]=1;80 myflag.counts++;83break;84 }85 }86for(index=0;index<B;index++){87if(myflag.flags[index]==0)88break;89 }90return index;91 }929394//下⼀个页⾯if在物理块中返回index,反-195int inBlock(int which){96//int i=0;97//while(i<B)98// if(block[i++]==which)99// return i-1;100for(int i=0;i<B;i++){101if(block[i]==which)102return i;103 }104return -1;105 }106107//打印⼀元组108void disPlay(){109int i=0;110while(i<B){111 printf("%d\t",block[i++]);112 }113 printf("\n");114return;115 }上⾯是博主使⽤C++(基本是C语法)编写的代码,运⾏结果如下://////////////////////////////////////////////////////////////////////////begin:缺页:(null)-->1缺页:(null)-->2缺页:(null)-->31 2 3缺页:3--被替换为-->41 2 41 2 41 2 4缺页:4--被替换为-->51 2 51 2 51 2 5缺页:1--被替换为-->33 2 5缺页:3--被替换为-->44 2 54 2 5end!//////////////////////////////////////////////////////////////////////////先进先出算法:先进先出置换算法(first in first out,FIFO)是淘汰最先进⼊内存的页⾯,即选择在内存中驻留时间最长的页⾯进⾏淘汰的算法。
- 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
- 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
- 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。
• • • • • • • • • • • • • • • • • •
p=freepf_head->next; /*按方式调新页面入内存页面*/ freepf_head->next=NUL; freepf_head->pn=page[i]; pl[page[i]].pfn=freepf_head->pfn; if(busypf_tail==NUL) busypf_head=busypf_tail=freepf_head; else { busypf_tail->next=freepf_head; busypf_tail=freepf_head; } freepf_head=p; } } printf("FIFO:%6.4F",1-(float)diseffect/320); } void LRU(int total_pf) {
• • • • • • • • • • • • • • • • • •
freepf_head->next=NUL; } pl[page[i]].pfn=freepf_head->pfn; pl[page[i]].time=present_time; freepf_head=freepf_head->next; } else pl[page[i]].time=present_time; present_time++; } printf("LRU:%6.4f",1-(float)diseffect/320); } void NUR(int total_pf) { int i,j,dp,cont_flag,old_dp; pfc_type *t; initialize(total_pf); dp=0;
• 操作系统页面置换算法代码 • • • • • • • • • • • • • • • • • #include <stdio.h>[1] #include <stdlib.h> #include <unistd.h> #define TRUE 1 #define FALSE 0 #define INVALID -1 #define NUL 0 #define total_instruction 320 /*指令流长*/ #define total_vp 32 /*虚页长*/ #define clear_period 50 /*清零周期*/ typedef struct{ /*页面结构*/ int pn,pfn,counter,time; }pl_type; pl_type pl[total_vp]; /*页面结构数组*/ struct pfc_struct{ /*页面控制结构*/ int pn,pfn; struct pfc_struct *next; };
页面置换算法
在地址映射过程中,若在页面中发现所要访问的页面不再内存中,则产生缺页 中断。当发生缺页中断时操作系统必须在内存选择一个页面将其移出内存,以 便为即将调入的页面让出空间。而用来选择淘汰哪一页的规则叫做页面置换算法 常见的置换算法有: 1.最佳置换算法(OPT)(理想置换算法) 2.先进先出置换算法(FIFO): 3 3.最近最久未使用(LRU)算法 LRU 4.Clock置换算法(LRU算法的近似实现) 5.最少使用(LFU)置换算法 6.工作集算法 7 . 工作集时钟算法 8. 老化算法(非常类似LRU的有效算法) 9. NRU(最近未使用)算法 10. 第二次机会算法
• • • • • • • • • • • • • • • • • • •
typedef struct pfc_struct pfc_type; pfc_type pfc[total_vp],*freepf_head,*busypf_head,*busypf_tail; int diseffect,a[total_instruction]; int page[total_instruction], offset[total_instruction]; void initialize(int); void FIFO(int); void LRU(int); void NUR(int); int main() { int S,i; srand((int)getpid()); S=(int)rand()%390; for(i=0;i<total_instruction;i+=1) /*产生指令队列*/ { a[i]=S; /*任选一指令访问点*/ a[i+1]=a[i]+1; /*顺序执行一条指令*/ a[i+2]=(int)rand()%390; /*执行前地址指令m’*/ a[i+3]=a[i+2]+1; /*执行后地址指令*/
• { • int i; • pfc_type *p, *t; • initialize(total_pf); /*初始化相关页面控制用数据结构*/ • busypf_head=busypf_tail=NUL; /*忙页面队列头,对列尾链接*/ • for(i=0;i<total_instruction;i++) • { • if(pl[page[i]].pfn==INVALID) /*页面失效*/ • { • diseffect+=1; /*失效次数*/ • if(freepf_head==NUL) /*无空闲页面*/ • { • p=busypf_head->next; • pl[busypf_head->pn].pfn=INVALID; /*释放忙页面队列中的第一 个页面*/ • freepf_head=busypf_head; • freepf_head->next=NUL; • busypf_head=p; • }
• int min,minj,i,j,present_time; • initialize(total_pf);present_time=0; • for(i=0;i<total_instruction;i++) • { • if(pl[page[i]].pfn==INVALID) /*页面失效*/ • { • diseffect++; • if(freepf_head==NUL) /*无空闲页面*/ • { • min=32767; • for(j=0;j<total_vp;j++) • if(min>pl[j].time&&pl[j].pfn!=INVALID) • { • min=pl[j].time; • minj=j; • } • freepf_head=&pfc[pl[minj].pfn]; • pl[minj].pfn=INVALID; • pl[minj].time=-1;
• • • • • • • • • • • • • • • • • • •
} freepf_head=&pfc[pl[dp].pfn]; pl[dp].pfn=INVALID; freepf_head->next=NUL; } pl[page[i]].pfn=freepf_head->pfn; freepf_head=freepf_head->next; } else pl[page[i]].counter=1; if(i%clear_period==0) for(j=0;j<total_vp;j++) pl[j].counter=0; } printf("NUR:%6.4f",1-(float)diseffect/320); } void initialize(int total_pf) /*初始化相关数据结构*/ /*用户进程的内存页面数*/ {
• for(i=0;i<total_instruction;i++) • { • if(pl[page[i]].pfn==INVALID) /*页面失效*/ • { • diseffect++; • if(freepf_head==NUL) /*无空闲页面*/ • { • cont_flag=TRUE;old_dp=dp; • while(cont_flag) • if(pl[dp].counter==0&&pl[dp].pfn!=INVALID) • cont_flag=FALSE; • else • { • dp++; • if(dp==total_vp) • dp=0; • if(dp==old_dp) • for(j=0;j<total_vp;j++) • pl[j].co• • • • • •
int i; diseffect=0; for(i=0;i<total_vp;i++) { pl[i].pn=i;pl[i].pfn=INVALID; /*置页面控制结构中的页号,页面为 空*/ pl[i].counter=0;pl[i].time=-1; /*页面控制结构中的访问次数为0, 时间为-1*/ } for(i=1;i<total_pf;i++) { pfc[i-1].next=&pfc[i];pfc[i-1].pfn=i-1;/*建立pfc[i-1]和pfc[i]之间的连 接*/ } pfc[total_pf-1].next=NUL;pfc[total_pf-1].pfn=total_pf-1; freepf_head=&pfc[0]; /*页面队列的头指针为pfc[0]*/ }
• S=(int)rand()%390; • } • for(i=0;i<total_instruction;i++) /*将指令序列变换成页地址流*/ • { • page[i]=a[i]/10; • offset[i]=a[i]%10; • } • for(i=4;i<=32;i++) /*用户内存工作区从4个页面到32个页面*/ • { • printf("%2d page frames",i); • FIFO(i); • LRU(i); • NUR(i); • printf("\n"); • } • return 0; • } • void FIFO(int total_pf) /*FIFO(First in First out)ALGORITHM*/ • /*用户进程的内存页面数*/