计算机操作系统:实验3 【虚拟存储器管理】

news2025/7/19 19:11:33

计算机操作系统:实验3 【虚拟存储器管理】

文章目录

  • 计算机操作系统:实验3 【虚拟存储器管理】
    • 一、前言
    • 二、实验目的
    • 三、实验环境
    • 四、实验内容
    • 五、实验说明
      • 1、设计中虚页和实页的表示
      • 2、关于缺页次数的统计
      • 3、LRU算法中“最近最久未用”页面的确定
      • 4、算法中实页的组织
    • 六、实验步骤
      • 1、理解好相关实验说明
      • 2、根据实验说明,画出相应的程序流程图
      • 3、按照程序流程图,用C语言编程并实现

一、前言

本期操作系统实验我们将了解学习如何使用C/C++实现虚拟存储器管理,并掌握FIFO和LRU算法进行页面置换的方法。

二、实验目的

请求页式虚存管理是常用的虚拟存储管理方案之一。通过请求页式虚存管理中对页面置换算法的模拟,有助于理解虚拟存储技术的特点,并加深对请求页式虚存管理的页面调度算法的理解。

三、实验环境

Turbo C 2.0/3.0或VC++6.0

我所使用的编译器是:Embarcadero Dev-C++

四、实验内容

本实验要求使用C语言编程模拟一个拥有若干个虚页的进程在给定的若干个实页中运行、并在缺页中断发生时分别使用FIFO和LRU算法进行页面置换的情形。其中虚页的个数可以事先给定(例如10个),对这些虚页访问的页地址流(其长度可以事先给定,例如20次虚页访问)可以由程序随机产生,也可以事先保存在文件中。要求程序运行时屏幕能显示出置换过程中的状态信息并输出访问结束时的页面命中率。程序应允许通过为该进程分配不同的实页数,来比较两种置换算法的稳定性。

五、实验说明

1、设计中虚页和实页的表示

本设计利用C语言的结构体来描述虚页和实页的结构。

在这里插入图片描述

在虚页结构中,pn代表虚页号,因为共10个虚页,所以pn的取值范围是0—9。pfn代表实页号,当一虚页未装入实页时,此项值为-1;当该虚页已装入某一实页时,此项值为所装入的实页的实页号pfn。time项在FIFO算法中不使用,在LRU中用来存放对该虚页的最近访问时间。

在实页结构中中,pn代表虚页号,表示pn所代表的虚页目前正放在此实页中。pfn代表实页号,取值范围(0—n-1)由动态指派的实页数n所决定。next是一个指向实页结构体的指针,用于多个实页以链表形式组织起来,关于实页链表的组织详见下面第4点。

2、关于缺页次数的统计

为计算命中率,需要统计在20次的虚页访问中命中的次数。为此,程序应设置一个计数器count,来统计虚页命中发生的次数。每当所访问的虚页的pfn项值不为-1,表示此虚页已被装入某实页内,此虚页被命中,count加1。最终命中率=count/20*100%。

3、LRU算法中“最近最久未用”页面的确定

为了能找到“最近最久未用”的虚页面,程序中可引入一个时间计数器countime,每当要访问一个虚页面时,countime的值加1,然后将所要访问的虚页的time项值设置为增值后的当前countime值,表示该虚页的最后一次被访问时间。当LRU算法需要置换时,从所有已分配实页的虚页中找出time值为最小的虚页就是“最近最久未用”的虚页面,应该将它置换出去。

4、算法中实页的组织

因为能分配的实页数n是在程序运行时由用户动态指派的,所以应使用链表组织动态产生的多个实页。为了调度算法实现的方便,可以考虑引入free和busy两个链表:free链表用于组织未分配出去的实页,首指针为free_head,初始时n个实页都处于free链表中;busy链表用于组织已分配出去的实页,首指针为busy_head,尾指针为busy_tail,初始值都为null。当所要访问的一个虚页不在实页中时,将产生缺页中断。此时若free链表不为空,就取下链表首指针所指的实页,并分配给该虚页。若free链表为空,则说明n个实页已全部分配出去,此时应进行页面置换:对于FIFO算法要将busy_head 所指的实页从busy链表中取下,分配给该虚页,然后再将该实页插入到busy链表尾部;对于LRU算法则要从所有已分配实页的虚页中找出time值为最小的虚页,将该虚页从装载它的那个实页中置换出去,并在该实页中装入当前正要访问的虚页。

六、实验步骤

1、理解好相关实验说明

  • FIFO

    先进先出 (FIFO) 页面替换算法是最简单的页面替换算法。在这个算法中,操作系统在队列中跟踪内存中的所有页面,最旧的页面在队列的前面。当需要替换页面时,选择队列前面的页面进行删除。

    设容量为内存可以容纳的页数。令 set 为内存中的当前页集。

    1.开始遍历页面
     i) 如果设置的页面少于容量
       a) 将页面一页一页地插入到集合中,直到集合的大小达到容量或全部 处理页面请求
       b) 同时维护页面中的 排队执行先进先出
       c) 增量页面错误
     ii) 否则
       如果当前页面存在于集合中,则什么都不做
       否则
         a) 从队列中删除第一页因为它是第一个被输入的记忆
         b) 将队列中的第一页替换为字符串中的当前页
         c) 将当前页面存储在队列中
         d) 增加页面错误
    
    2.返回页面错误
    
  • LRU

    最近最少使用算法是一种贪心算法,其中要替换的页面是最近最少使用的。这个想法是基于引用的位置,最近最少使用的页面不太可能。

    设容量为页数 内存可以保存。让设置为当前内存中的页面集。

    1- 开始遍历页面.
     i) 如果设置的页面少于容量
       a) 将页面一页一页地插入到集合中,直到集合的大小达到容量或全部 处理页面请求
       b) 同时维护最近发生的地图中每个页面的索引称为索引
       c) 增量页面错误
     ii) 否则 
       如果当前页面存在于集合中,则什么都不做
       否则 
         a) 找到集合中最少的页面 最近用过。我们使用索引数组找到它。我们基本上需要将页面替换为最小指数。
         b) 用当前页面替换找到的页面
         c) 增加页面错误
         d) 更新当前页面的索引
    
    2. 返回页面错误
    

2、根据实验说明,画出相应的程序流程图

  • FIFO

在这里插入图片描述

  • LRU

    在这里插入图片描述

3、按照程序流程图,用C语言编程并实现

  • FIFO

    #include<bits/stdc++.h>
    using namespace std;
    
    // 使用 FIFO 查找页面错误的函数
    int pageFaults(int pages[], int n, int capacity)
    {
        /*
        表示一组当前页面。我们用一个 unordered_set 以便我们快速检查页面是否存在于集合中
        */
    	unordered_set<int> s;
    
    	// 以FIFO方式存储页面
    	queue<int> indexes;
    
    	// 从初始页面开始
    	int page_faults = 0;
    	for (int i=0; i<n; i++)
    	{
    		// 检查集合是否可以容纳更多页面
    		if (s.size() < capacity)
    		{
                /*
                如果不存在则将其插入集合 already表示页面错误
                */
    			if (s.find(pages[i])==s.end())
    			{
    				// 将当前页面插入集合
    				s.insert(pages[i]);
    
    				// 增量页面错误
    				page_faults++;
    
    				// 将当前页面推入队列
    				indexes.push(pages[i]);
    			}
    		}
    
    		/*
    		如果集合已满则需要执行 FIFO 即从中删除队列的第一页 设置和排队并插入当前页面
    		*/
    		else
    		{
    			/*
    			检查当前页面是否已经存在于集合中
    			*/
    			if (s.find(pages[i]) == s.end())
    			{
                    /*
                    将第一页存储在队列用于查找和从集合中删除页面
                    */
    				int val = indexes.front();
    				
    				// 从队列中弹出第一页
    				indexes.pop();
    
    				// 从集合中删除索引页
    				s.erase(val);
    
    				// 在集合中插入当前页面
    				s.insert(pages[i]);
    
    				// 将当前页面推入队列
    				indexes.push(pages[i]);
    
    				// 增量页面错误
    				page_faults++;
    			}
    		}
    	}
    
    	return page_faults;
    }
    
    
    int main()
    {
    	int pages[] = {7, 0, 1, 2, 0, 3, 0, 4,
    				2, 3, 0, 3, 2};
    	int n = sizeof(pages)/sizeof(pages[0]);
    	int capacity = 4;
    	cout << pageFaults(pages, n, capacity);
    	return 0;
    }
    

    运行的结果是:

    7
    
  • LRU

    #include<bits/stdc++.h>
    using namespace std;
    
    // 使用索引查找页面错误的函数
    int pageFaults(int pages[], int n, int capacity)
    {
    	// 表示一组当前页面。我们用个unordered_set 以便我们快速检查页面是否存在于集合中
    	unordered_set<int> s;
    
    	// 存储最近最少使用的索引页面数
    	unordered_map<int, int> indexes;
    
    	// 从初始页面开始
    	int page_faults = 0;
    	for (int i=0; i<n; i++)
    	{
    		// 检查集合是否可以容纳更多页面
    		if (s.size() < capacity)
    		{
    			// 如果不存在则将其插入集合already表示页面错误
    			if (s.find(pages[i])==s.end())
    			{
    				s.insert(pages[i]);
    
    				// 增量页面错误
    				page_faults++;
    			}
    
    			// 存储最近使用的索引每页
    			indexes[pages[i]] = i;
    		}
    
    		// 如果集合已满则需要执行lru即删除最近最少使用的页面并插入当前页
    		else
    		{
    			// 检查当前页面是否已经存在于集合中
    			if (s.find(pages[i]) == s.end())
    			{
    				// 查找最近最少使用的页面存在于集合中
    				int lru = INT_MAX, val;
    				for (auto it=s.begin(); it!=s.end(); it++)
    				{
    					if (indexes[*it] < lru)
    					{
    						lru = indexes[*it];
    						val = *it;
    					}
    				}
    
    				// 删除索引页
    				s.erase(val);
    
    				// 插入当前页
    				s.insert(pages[i]);
    
    				// 增量页面错误
    				page_faults++;
    			}
    
    			// 更新当前页面索引
    			indexes[pages[i]] = i;
    		}
    	}
    
    	return page_faults;
    }
    
    
    int main()
    {
    	int pages[] = {7, 0, 1, 2, 0, 3, 0, 4, 2, 3, 0, 3, 2};
    	int n = sizeof(pages)/sizeof(pages[0]);
    	int capacity = 4;
    	cout << pageFaults(pages, n, capacity);
    	return 0;
    }
    

    运行的结果是:

    6
    

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/8075.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

删除类及其对象的属性:delattr()函数

【小白从小学Python、C、Java】 【Python-计算机等级考试二级】 【Python-数据分析】 删除类及其对象的属性 delattr()函数 [太阳]选择题 请问对以下Python代码说法错误的是&#xff1f; class MyClass1: x 1 y 2 myObject1 MyClass1() print(【访问】myObject1的属…

Revit中“结构框架显示与剪切“的应用和一键剪切功能

一、Revit关于"结构框架显示与剪切"的应用 结构框架&#xff1a;顾名思义其实它表示的就是结构梁而已&#xff0c;但是我们画图的时候往往会显示"实线"和"虚线"&#xff0c;以至于在出结构图纸的时候&#xff0c;达不到出图要求 NO.2、应用 Part…

ISCTF

upload upload,一道phar文件上传题目 <?php class upload{public $filename;public $ext;public $size;public $Valid_ext;public function __construct(){$this->filename $_FILES["file"]["name"];$this->ext end(explode(".", …

[山东科技大学OJ]1490 Problem F: 该按哪些键

Time Limit: 1 Sec Memory Limit: 128 MB Submit: 1693 Solved: 433 [Submit][Status] Description Peter在手机上打字时一直用全键键盘来输入&#xff0c;但最近不知道怎么搞的&#xff0c;把全键键盘弄丢了&#xff0c;只剩下了9键键盘。一项喜欢高科技的Peter却不会用9键…

彻底搞懂nodejs事件循环

nodejs是单线程执行的&#xff0c;同时它又是基于事件驱动的非阻塞IO编程模型。这就使得我们不用等待异步操作结果返回&#xff0c;就可以继续往下执行代码。当异步事件触发之后&#xff0c;就会通知主线程&#xff0c;主线程执行相应事件的回调。 以上是众所周知的内容。今天…

ASEMI整流桥D3KB100参数,D3KB100规格,D3KB100封装

编辑-Z ASEMI整流桥D3KB100参数&#xff1a; 型号&#xff1a;D3KB100 最大重复峰值反向电压&#xff08;VRRM&#xff09;&#xff1a;1000V RMS反向电压VR(RMS)&#xff1a;700 平均整流输出电流&#xff08;IO&#xff09;&#xff1a;3A 峰值正向浪涌电流&#xff08…

【论文阅读】时序动作检测系列论文精读(2019年)

文章目录1. BMN: Boundary-Matching Network for Temporal Action Proposal Generation论文目的——拟解决问题贡献——创新实现流程详细方法2. MGG: Multi-granularity Generator for Temporal Action Proposal论文目的——拟解决问题贡献——创新实现流程详细方法3. P-GCN: G…

稳压二极管的应用及注意事项

文章目录稳压二极管也被称为齐纳二极管 齐纳二极管和普通二极管的伏安特性曲线 齐纳二极管的工作原理 稳压二极管的伏安特性曲线的正向特性和普通二极管差不多&#xff0c;反向特性是在反向电压低于反向击穿电压时&#xff0c;反向申阳很大&#xff0c;反向漏电流极小。但是…

【学习笔记】AGC028/AGC007

AGC028 Removing Blocks High Elements 好仙啊。 我会转化&#xff01;&#xff01;问题转化为在原序列剩下的数中取ISISIS序列aaa,bbb&#xff0c;满足cx∣a∣cy∣b∣cx|a|cy|b|cx∣a∣cy∣b∣ 。对于没在a,ba,ba,b序列中的数&#xff0c;可以通过恰当放置使其不对前缀最大…

并发编程- synchronized,Lock及volatile的使用

文章目录并发编程的可见性问题解决方法synchronizedLockvolatile并发编程的可见性问题 多线程访问共享变量&#xff0c;造成线程不安全&#xff0c;最后的数值不对 public class VDemo {private static int num 0;public static void add() {num;}public static void main(St…

红队内网渗透神器--CobaltStrike安装教程

CobaltStrike介绍&#xff1a; CobaltStrike是一款渗透测试神器&#xff0c;被业界人称为CS神器。CobaltStrike分为客户端与服务端&#xff0c;服务端是一个&#xff0c;客户端可以有多个&#xff0c;可被团队进行分布式协团操作。 CobaltStrike集成了端口转发、服务扫描&…

Ubuntu 手动配置DNS

使用ping命令测试百度域名时发现&#xff0c;无法解析这个域名&#xff0c;说明当前系统上没有配置DNS服务器。配置DNS服务器的方式主要有以下两种&#xff1a; 目录 1、修改DNS配置文件 /etc/resolv.conf 2、修改网卡配置文件 /etc/network/interfaces 1、修改DNS配置文件 /e…

【附源码】计算机毕业设计JAVA宠物云寄养系统

【附源码】计算机毕业设计JAVA宠物云寄养系统 目运行 环境项配置&#xff1a; Jdk1.8 Tomcat8.5 Mysql HBuilderX&#xff08;Webstorm也行&#xff09; Eclispe&#xff08;IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持&#xff09;。 项目技术&#xff1a; JAVA myba…

Vue3 - toRef() 使用教程

介绍 它可用于为响应式对象上的 property 创建 ref&#xff0c;这样创建的 ref 与其源 property 保持同步&#xff0c;当改变源 property 时&#xff0c;将更新 ref &#xff0c;反之亦然。 这段话有些晦涩难懂&#xff0c;其实用大白话说&#xff0c;就是当你使用 reactive 创…

如何拆分PDF成单页?这三个方法分享给你

很多朋友在平时的工作中&#xff0c;经常需要处理一些PDF格式的文件&#xff0c;但是如果PDF文件的占用空间太大&#xff0c;难以进行操作处理&#xff0c;这时我们就需要先将其拆分成多个小文件&#xff0c;那你们知道要怎么把PDF拆分成多个文件吗&#xff1f;今天我就来给大家…

RK3568平台开发系列讲解(LCD篇)DRM 显示框架

🚀返回专栏总目录 文章目录 一、DRM 显示框架二、DRM 驱动和 libdrm 交互过程2.1、GEM:2.2、KMS:三、DRM 驱动路径3.1、Uboot驱动路径3.2、内核驱动路径沉淀、分享、成长,让自己和他人都能有所收获!😄 📢DRM 英文名叫 Direct Rendering Manager,用来管理显示输出,图…

App Languages 批量导入管理flutter多语言文案

前段时间AppLanguages推出了iOS、Mac版的多语言文案导入功能&#xff0c;好几个小伙伴点赞&#xff0c;称其为“干货工具”&#xff0c;最近加班加点支持了flutter的多语言文案管理功能。 操作界面 批量导入 1&#xff09;需要选择lib文件夹的路径&#xff0c;方便创建和寻找…

2022年11月华南师范大学自考本科-计算机信息管理课程实验—《数据库系统原理》实践题目

《 计算机信息管理课程实验——数据库系统原理 》课程试卷 答卷提交说明&#xff1a; 在mysql环境下填写SQL命令完成以下实践的题目&#xff0c;并返回执行结果的截图&#xff0c;答卷的答题格式如下&#xff0c;包括三部分&#xff1a;题目&#xff0c;SQL文本代码&#x…

【测试开发面试】6家企业真实面试,最终成功入职外企......

目录&#xff1a;导读前言一、Python编程入门到精通二、接口自动化项目实战三、Web自动化项目实战四、App自动化项目实战五、一线大厂简历六、测试开发DevOps体系七、常用自动化测试工具八、JMeter性能测试九、总结&#xff08;尾部小惊喜&#xff09;前言 粉丝小A 测试开发的…

3、HTML——注释、转义字符、超链接标签、锚链接、功能性超链接、列表标签、有序列表、无序列表、定义列表

目录 一、注释标签 二、转义字符 1、空格&#xff1a; 2、大于号/小于号&#xff1a;>/< 3、引号&#xff1a;" 4、版权&#xff1a;© 5、商标&#xff1a;& 6、常见转义字符 三、超链接标签&#xff1a;a 四、锚链接 1、跳转同网页位置 2、…