操作系统实验五 存储管理

这篇具有很好参考价值的文章主要介绍了操作系统实验五 存储管理。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

★观前提示本篇内容为操作系统实验内容,代码等内容经测试没有问题,但是可能会不符合每个人实验的要求,因此以下内容建议仅做思路参考

一、实验目的
  1. 掌握虚拟内存的管理机制。
  2. 了解虚拟存储技术的特点
  3. 掌握请求分页存储管理的页面置换算法
二、实验内容

设计一个虚拟存储区和内存工作区,并使用以下置换算法计算访问命中率:先进先出页面置换算法、最近最少使用页面置换算法、最佳淘汰页面置换算法等。

  1. 通过随机数产生一个指令序列,共320条指令。

  2. 指令序列变换为页地址流

  3. 设计一个虚拟存储区和内存工作区,并使用下列各种算法计算,当分配给作业的内存块数从4——32时,访问的命中率各是多少,并进行比较。

    ①先进先出的算法(FIFO);
    ②最近最少使用算法(LRR);
    ③最佳淘汰算法(OPT),先淘汰最不常用的页地址;
    ④最少访问页面算法(LFR);(选做)
    ⑤最近最不经常使用算法(NUR)。(选做)
    其中①、②和③为必做内容,④和⑤为选做内容。

三、具体实现

1️⃣实验方案的选择
①先进先出的算法(FIFO);
②最近最少使用算法(LRR);
③最佳淘汰算法(OPT),先淘汰最不常用的页地址;
④最少访问页面算法(LFR);
⑤最近最不经常使用算法(NUR)。

本次实验共实现以上五种页面置换算法,查询了书籍和相关资料之后,理解了几个算法的含义、以及它们的特点,
另外在老师所发文件源码的基础上,借助C++语言对以上五种算法进行了实现。详细步骤如下:

①创建test5.cpp文件
操作系统实验五 存储管理

②编程如下

#include<iostream>
#include<ctime>
#include<random>

using namespace std;

#define TRUE 1
#define FALSE 0
#define INVALID -1
//#define NULL 0

#define total_instruction 320 //指令数量
#define total_vp 32 //页表数量
#define clear_period 50 //清0周期

struct pl_type { /*页表结构*/
	int pn, pfn, counter, time;
};
pl_type pl[total_vp]; //页表数组

struct pfc_type { /*内存表结构*/
	int pn, pfn;
	pfc_type* next;
};
pfc_type pfc[total_vp], * freepf_head, * busypf_head, * busypf_tail;

int diseffect; // 未命中次数
int a[total_instruction]; // 存储320条指令
int page[total_instruction]; // 每条指令对应的页表号
int offset[total_instruction];

int initialize(int);
int FIFO(int);
int LRU(int);
int NUR(int);
int LFU(int);
int OPT(int);

int main() {
	int s;
	srand(unsigned(time(0)));
	for (int i = 0; i < total_instruction; i += 4) { /*产生指令队列*/
		s = rand() % 320;
		a[i] = s + 1; //顺序执行一条指令
		a[i + 1] = rand() % (a[i] + 1); //执行前地址指令m'
		a[i + 2] = a[i + 1] + 1; //顺序执行一条指令
		a[i + 3] = rand() % (319 - a[i + 2]) + (a[i + 2] + 1); // 执行后地址指令

		if (a[i] > 319 || a[i + 1] > 319 || a[i + 2] > 319 || a[i + 3] > 319) {
			i -= 4;
		}
	}
	for (int i = 0; i < total_instruction; i++) { /*将指令序列变换成页表地址流*/
		page[i] = a[i] / 10;
		offset[i] = a[i] % 10;
	}
	for (int i = 4; i <= 32; i++) { /*用户内存工作区从4个页面到32个页面*/
		printf("---%2d page frames---\n", i);
		FIFO(i);
		LRU(i);
		NUR(i);
		LFU(i);
		OPT(i);
	}
	return 0;
}

int initialize(int total_pf) { /*初始化相关数据结构*/
	diseffect = 0;
	// 初始化页表
	for (int i = 0; i < total_vp; i++) {
		pl[i].pn = i;
		pl[i].pfn = INVALID; // 初始,该页不在内存中
		pl[i].counter = 0; // 访问次数或作为引用位
		pl[i].time = -1; // 时间
	}
	// 初始内存表,建立pfc[i-1]和pfc[i]之间的链接
	for (int i = 0; i < total_pf - 1; i++) {
		pfc[i].next = &pfc[i + 1];
		pfc[i].pfn = i;
	}
	pfc[total_pf - 1].next = NULL;
	pfc[total_pf - 1].pfn = total_pf - 1;
	freepf_head = &pfc[0]; //内存空页面队列的头指针为pfc[0]

	return 0;
}

int FIFO(int total_pf) { /*先进先出算法*/
	pfc_type* p;
	initialize(total_pf); //初始化相关页面控制用数据结构
	busypf_head = busypf_tail = NULL; //内存页的队列头,队列尾指针接
	for (int i = 0; i < total_instruction; i++) {
		if (pl[page[i]].pfn == INVALID) { //页表项不在内存中
			diseffect += 1; //失效次数
			if (freepf_head == NULL) { //内存无空闲页面
				p = busypf_head->next;
				pl[busypf_head->pn].pfn = INVALID;
				freepf_head = busypf_head; //释放忙页面队列的第一个页面
				freepf_head->next = NULL;
				busypf_head = p;
			}
			// 按FIFO方式调新页面入内存页面
			p = freepf_head->next; // 先保存内存表中当前位置的下一位置
			freepf_head->next = NULL;
			freepf_head->pn = page[i]; // 页表号
			pl[page[i]].pfn = freepf_head->pfn; // 内存块号

			if (busypf_tail == NULL) {
				// busypf_head指向最老的,busypf_tail指向最新的
				busypf_head = busypf_tail = freepf_head;
			}
			else {
				busypf_tail->next = freepf_head; //free页面减少一个
				busypf_tail = freepf_head;
			}
			freepf_head = p;
		}
	}
	printf("FIFO:%6.4f\n", 1 - diseffect / 320.0);

	return 0;
}

int LRU(int total_pf) { /*最近最久未使用算法(使用时钟计数器)*/
	int min, minj, present_time;
	initialize(total_pf);
	present_time = 0;
	for (int i = 0; i < total_instruction; i++) {
		if (pl[page[i]].pfn == INVALID) { //页面失效,不在内存中
			diseffect++;
			if (freepf_head == NULL) { //内存无空闲页面
				min = 32767;
				for (int j = 0; j < total_vp; j++) { //找出内存块中time的最小值
					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->next = NULL;
			}
			pl[page[i]].pfn = freepf_head->pfn; //有空闲页面,改为有效(内存块号)
			pl[page[i]].time = present_time;
			freepf_head = freepf_head->next; //减少一个free 页面
		}
		else {
			pl[page[i]].time = present_time; //命中则设置时间
		}
		present_time++;
	}
	printf("LRU:%6.4f\n", 1 - diseffect / 320.0);
	return 0;
}

int NUR(int total_pf) { /*最近未使用算法(每执行50条指令引用位清零一次)*/
	int dp, cont_flag, old_dp;
	initialize(total_pf);
	dp = 0;
	for (int i = 0; i < total_instruction; i++) {
		if (pl[page[i]].pfn == INVALID) { //页面失效,不在内存中
			diseffect++;
			if (freepf_head == NULL) { //无空闲页面
				cont_flag = TRUE;
				old_dp = dp;
				while (cont_flag) {
					// 查页表(对应引用位0,在内存中)
					if (pl[dp].counter == 0 && pl[dp].pfn != INVALID) {
						cont_flag = FALSE;
					}
					else {
						dp++;
						if (dp == total_vp) {
							dp = 0;
						}
						if (dp == old_dp) {
							// 第2轮扫描结束,失败(引用位全部置为0)
							for (int j = 0; j < total_vp; j++) {
								pl[j].counter = 0;
							}
						}
					}
				}
				freepf_head = &pfc[pl[dp].pfn]; //腾出一个单元(对应的内存块)
				pl[dp].pfn = INVALID;
				freepf_head->next = NULL;
			}
			pl[page[i]].pfn = freepf_head->pfn;
			freepf_head = freepf_head->next;
		}
		else {
			pl[page[i]].counter = 1; // 在内存中,“引用位”置为1
		}
		// 每执行50条指令,引用位清零一次
		if (i % clear_period == 0) {
			for (int j = 0; j < total_vp; j++) {
				pl[j].counter = 0;
			}
		}
	}
	printf("NUR:%6.4f\n", 1 - diseffect / 320.0);

	return 0;
}

int OPT(int total_pf) { /*最佳置换算法*/
	int max, maxpage, d, dist[total_vp];
	initialize(total_pf);
	for (int i = 0; i < total_instruction; i++) {
		if (pl[page[i]].pfn == INVALID) { //页面失效,不在内存中
			diseffect++;
			if (freepf_head == NULL) { //无空闲页面
				for (int j = 0; j < total_vp; j++) {
					if (pl[j].pfn != INVALID) {
						dist[j] = 32767; /* 最大"距离" */
					}
					else {
						dist[j] = 0;
					}
				}
				d = 1;
				for (int j = i + 1; j < total_instruction; j++) {
					if (pl[page[j]].pfn != INVALID && dist[page[j]] == 32767) {
						dist[page[j]] = d;
					}
					d++;
				}
				max = -1;
				for (int j = 0; j < total_vp; j++) {
					if (max < dist[j]) {
						max = dist[j];
						maxpage = j;
					}
				}
				freepf_head = &pfc[pl[maxpage].pfn];
				freepf_head->next = NULL;
				pl[maxpage].pfn = INVALID;
			}
			pl[page[i]].pfn = freepf_head->pfn;
			freepf_head = freepf_head->next;
		}
	}
	printf("OPT:%6.4f\n", 1 - diseffect / 320.0);

	return 0;
}

int LFU(int total_pf) { /*最不经常使用置换法*/
	int i, j, min, minpage;
	initialize(total_pf);
	for (i = 0; i < total_instruction; i++) {
		if (pl[page[i]].pfn == INVALID) { //页面失效,不在内存中
			diseffect++;
			if (freepf_head == NULL) { //无空闲页面
				min = 32767;
				for (j = 0; j < total_vp; j++) {
					if (min > pl[j].counter && pl[j].pfn != INVALID) {
						min = pl[j].counter;
						minpage = j;
					}
					//pl[j].counter = 0;
					//pl[j].counter >>= 1;
				}
				freepf_head = &pfc[pl[minpage].pfn];
				pl[minpage].pfn = INVALID;
				freepf_head->next = NULL;
			}
			pl[page[i]].pfn = freepf_head->pfn; //有空闲页面,改为有效
			pl[page[i]].counter++;
			freepf_head = freepf_head->next; //减少一个free页面
		}
		else {
			pl[page[i]].counter++; // 软件计数器(被访问次数)
		}
		// 定期右移
		if (i % 15 == 0) {
			for (int j = 0; j < total_vp; j++) {
				pl[j].counter >>= 1;
			}
		}
	}
	printf("LFU:%6.4f\n", 1 - diseffect / 320.0);

	return 0;
}

③编程完毕之后,借助"wq"进行保存并退出
操作系统实验五 存储管理

④对test5.cpp进行编译
操作系统实验五 存储管理

⑤编译完成后,对实验结果进行输出
操作系统实验五 存储管理

2️⃣实验分析

在内存页面数较少(4 ~ 5页)时,五种算法的命中率差别不大,都是30%左右。在内存页面为7 ~ 18个页面之间时,5种算法的访内命中率大致在35% ~ 60%之间变化。但是,FIFO算法与OPT算法之间的差别一般在6 ~ 10个百分点左右。在内存页面为25~32个页面时,由于用户进程的所有指令基本上都已装入内存,使命中率增加,从而算法之间的差别不大。比较上述5种算法,以OPT算法的命中率最高,NUR算法次之,再就是LFU算法和LRU算法,其次是FIFO算法。就本问题,在15页之前,FIFO的命中率比LRU的高。

四、实验总结

① 掌握了虚拟内存的管理机制。
②了解虚拟存储技术的特点,可以大大提高存储系统整体访问带宽,为存储资源管理提供了更好的灵活性。
③ 掌握了请求分页存储管理的页面置换算法, FIFO、LRR、OPT等等

2022.9.21记录:Code_流苏(CSDN)
如有任何疑问,评论回复,看到即回,欢迎大家多多交流学习!
★以上实验内容仅供参考。文章来源地址https://www.toymoban.com/news/detail-460601.html

到了这里,关于操作系统实验五 存储管理的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 操作系统的存储管理

      文件系统是操作系统用于明确存储设备或分区上的文件的方法和数据结构;即在存储设备上组织文件的方法。操作系统中负责管理和存储文件信息的软件机构称为文件管理系统,简称文件系统。 注意:拷的文件比较大时,需要格式化为NTFS 注意:一个硬盘只有一个扩展分区

    2024年01月24日
    浏览(36)
  • 【操作系统】存储器管理练习

    12.(考研真题) 假设一个分页存储系统具有 快表 ,多数活动页表项都可以存在于其中。 若页表放在内存中,内存访问时间是1ns,快表的命中率是85%,快表的访问时间为0.1ns, 则 有效存取时间 为多少? 15.  已知某分页系统,内存容量为64KB,页面大小为1KB,对一个4页大的作业

    2024年02月05日
    浏览(95)
  • 操作系统 | 实验八 文件管理

    掌握文件的存取方法;掌握文件的逻辑结构和物理结构;掌握存储空间的分配和回收;掌握磁盘管理与调度。 用程序模拟磁盘的调度过程,并计算各磁盘调度算法包括先来先服务算法、最短寻道时间优先算法、扫描算法和循环扫描算法的平均寻道长度。 本实验是模拟操作系

    2024年02月06日
    浏览(46)
  • 操作系统实验之文件管理

    目录 一、实验目的 二、实验内容 三、实验思路 四、主要数据结构 五、实验流程图 六、实现代码 七、运行结果 通过这次实验,掌握文件系统的用户管理,掌握普通文件、目录文件管理的基本原理。 1、通过初始化操作建立一个模拟外存空间的虚拟磁盘文件,在该文件中保存

    2024年02月05日
    浏览(47)
  • 【操作系统】——基本分页存储管理

    将内存分为一个个大小相等的分区, 这些分区称作为(页框、页帧、内存块、物理块、物理页面)若对分区进从编号,则又有了对应的(页框号、页帧号、内存块号、物理块号、物理页号),从0开始 进程的信息都是要存在内存中的,既然内存有了分区,那么进程逻辑地址空间

    2024年02月06日
    浏览(43)
  • 操作系统实验——进程管理的算法实现

    笔者在大学下属的事业单位上班,最近去帮着带下操作系统的实验课,这里随手水点参考代码,欢迎各位领导老师莅临指正 编写一个简单的进程调度器 进程控制块(PCB)的定义与管理 进程调度算法的实现 进程创建、销毁和切换 给定一批进程对比3-4种调度算法的时间(自选

    2024年02月06日
    浏览(44)
  • Linux操作系统实验三 文件管理(一)

      1.实验目的与要求 了解Linux文件系统目录结构 掌握目录管理的相关操作 掌握文件管理的相关操作 2.实验平台 实验室安装的实验环境(Linux操作系统)和头歌(www.educoder.net)实验平台(课程实验) 3.实验内容 文件系统目录结构理论知识练习 linux 下目录的创建、应用、查看、

    2024年02月03日
    浏览(52)
  • 计算机系统结构与操作系统实验三(6)-内存管理

    实现内存管理 这里修改makefile文件和run.sh文件 在《操作系统真相还原源码》的基础上稍加修改makefile 注意:这里要用 make all 命令来执行makefile文件了 本实验所有源码👉👉👉 计算机系统结构与操作系统实验三bochs源代码

    2024年02月15日
    浏览(43)
  • 【操作系统笔记04】操作系统之内存管理方式(分页、分段、段页式)、虚拟存储技术、页面置换算法

    这篇文章,主要介绍操作系统之内存管理方式(分页、分段、段页式)、虚拟存储技术、页面置换算法。 目录 一、操作系统 1.1、基地址变换机构 1.2、具有快表的地址变换机构

    2023年04月21日
    浏览(42)
  • 【Linux操作系统】【综合实验三 用户帐号、文件系统与系统安全管理】

    要求掌握Linux系统用户的创建、删除与管理操作;熟悉Linux文件系统的管理模式,学会创建用户文件系统并装载和卸载文件系统;掌握超级用户的管理方式与权限,并实施对普通用户的管理;熟悉Linux系统安全机制与相关管理方法。 通过这个第三阶段实验,要求掌握以下操作与

    2023年04月14日
    浏览(35)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包