操作系统课程设计-内存管理

目录

前言

1 实验题目

2 实验目的

3 实验内容

3.1 步骤

3.2 关键代码

3.2.1 显示虚拟内存的基本信息

3.2.2 遍历当前进程的虚拟内存

4 实验结果与分析

5 代码


前言

         本实验为课设内容,博客内容为部分报告内容,仅为大家提供参考,请勿直接抄袭,另外,本次实验所用平台是dev c++5.11

1 实验题目

        实验五 内存管理

2 实验目的

        了解 Windows 的内存结构和虚拟内存的管理,理解进程的虚拟内存空间和物理内存的映射关系。加深对操作系统内存管理、虚拟存储管理等理论知识的理解。

3 实验内容

3.1 步骤

        (1)步骤1:打开Dev-C++5.11 新建一个文件,命名为实验5,并保存为cpp文件。

        (2)步骤2:将清单6-1的源代码复制到实验5.cpp文件中,并点击编译按钮将其编译成可执行文件,再进入到保存文件的目录下,右键,选择在终端打开,运行编译好的可执行文件,观察运行结果。

3.2 关键代码

3.2.1 显示虚拟内存的基本信息

// 首先,让我们获得系统信息SYSTEM_INFO si;:: ZeroMemory(&si, sizeof(si) ) ;:: GetSystemInfo(&si) ;
// 使用外壳辅助程序对一些尺寸进行格式化TCHAR szPageSize[MAX_PATH];::StrFormatByteSize(si.dwPageSize, szPageSize, MAX_PATH) ;DWORD dwMemSize = (DWORD64)si.lpMaximumApplicationAddress -(DWORD64) si.lpMinimumApplicationAddress;TCHAR szMemSize [MAX_PATH] ;:: StrFormatByteSize(dwMemSize, szMemSize, MAX_PATH) ;
// 将内存信息显示出来std :: cout << "Virtual memory page size: " << szPageSize << std :: endl;std :: cout.fill ('0') ;std :: cout << "Minimum application address: 0x"<< std :: hex << std :: setw(8)<< (DWORD64) si.lpMinimumApplicationAddress<< std :: endl;std :: cout << "Maximum application address: 0x"<< std :: hex << std :: setw(8)<< (DWORD64) si.lpMaximumApplicationAddress<< std :: endl;std :: cout << "Total available virtual memory: "<< szMemSize << std :: endl ;

3.2.2 遍历当前进程的虚拟内存

// 首先,获得系统信息SYSTEM_INFO si;:: ZeroMemory(&si, sizeof(si) ) ;:: GetSystemInfo(&si) ;
// 分配要存放信息的缓冲区MEMORY_BASIC_INFORMATION mbi;:: ZeroMemory(&mbi, sizeof(mbi) ) ;
// 循环整个应用程序地址空间LPCVOID pBlock = (LPVOID) si.lpMinimumApplicationAddress;while (pBlock < si.lpMaximumApplicationAddress) {
// 获得下一个虚拟内存块的信息if (:: VirtualQueryEx(hProcess, // 相关的进程pBlock, // 开始位置&mbi, // 缓冲区sizeof(mbi))==sizeof(mbi) ) { // 大小的确认
// 计算块的结尾及其大小LPCVOID pEnd = (PBYTE) pBlock + mbi.RegionSize;TCHAR szSize[MAX_PATH];:: StrFormatByteSize(mbi.RegionSize, szSize, MAX_PATH) ;
// 显示块地址和大小std :: cout.fill ('0') ;std :: cout<< std :: hex << std :: setw(8) << (DWORD64) pBlock<< "-"<< std :: hex << std :: setw(8) << (DWORD64) pEnd<< (:: strlen(szSize)==7? " (" : " (") << szSize<< ") " ;
// 显示块的状态switch(mbi.State) {case MEM_COMMIT :std :: cout << "Committed" ;break;case MEM_FREE :std :: cout << "Free" ;break;case MEM_RESERVE :std :: cout << "Reserved" ;break;}
// 显示保护if(mbi.Protect==0 && mbi.State!=MEM_FREE) {mbi.Protect=PAGE_READONLY;}ShowProtection(mbi.Protect);
// 显示类型switch(mbi.Type) {case MEM_IMAGE :std :: cout << ", Image" ;break;case MEM_MAPPED:std :: cout << ", Mapped";break;case MEM_PRIVATE :std :: cout << ", Private" ;break;}

4 实验结果与分析

(1)运行程序后的部分结果如下图所示:

图1.1 内存管理部分结果

(2)由上面的部分结果可知,虚拟内存每页容量为4KB,最小应用地址为0x00010000,最大应用地址为0x7ffffffeffff,当前可供应用程序使用的内存空间为3.99GB,当前计算机实际内存大小为16GB,理论上每个Windows应用程序可以独占的最大存储空间是19.99GB。

5 代码

// 工程 vmwalker
#include <windows.h>
#include <iostream>
#include <shlwapi.h>
#include <iomanip>
#include<bits/stdc++.h>
#include<stdio.h>
#include<limits.h>
#pragma comment(lib, "Shlwapi.lib")// 以可读方式对用户显示保护的辅助方法。
// 保护标记表示允许应用程序对内存进行访问的类型
// 以及操作系统强制访问的类型
inline bool TestSet(DWORD dwTarget, DWORD dwMask) {return ((dwTarget &dwMask) == dwMask) ;
}
# define SHOWMASK(dwTarget, type) \if (TestSet(dwTarget, PAGE_##type) ) \{std :: cout << ", " << #type; }
void ShowProtection(DWORD dwTarget) {SHOWMASK(dwTarget, READONLY) ;SHOWMASK(dwTarget, GUARD) ;SHOWMASK(dwTarget, NOCACHE) ;SHOWMASK(dwTarget, READWRITE) ;SHOWMASK(dwTarget, WRITECOPY) ;SHOWMASK(dwTarget, EXECUTE) ;SHOWMASK(dwTarget, EXECUTE_READ) ;SHOWMASK(dwTarget, EXECUTE_READWRITE) ;SHOWMASK(dwTarget, EXECUTE_WRITECOPY) ;SHOWMASK(dwTarget, NOACCESS) ;
}
// 遍历整个虚拟内存并对用户显示其属性的工作程序的方法
void WalkVM(HANDLE hProcess) {
// 首先,获得系统信息SYSTEM_INFO si;:: ZeroMemory(&si, sizeof(si) ) ;:: GetSystemInfo(&si) ;
// 分配要存放信息的缓冲区MEMORY_BASIC_INFORMATION mbi;:: ZeroMemory(&mbi, sizeof(mbi) ) ;
// 循环整个应用程序地址空间LPCVOID pBlock = (LPVOID) si.lpMinimumApplicationAddress;while (pBlock < si.lpMaximumApplicationAddress) {
// 获得下一个虚拟内存块的信息if (:: VirtualQueryEx(hProcess, // 相关的进程pBlock, // 开始位置&mbi, // 缓冲区sizeof(mbi))==sizeof(mbi) ) { // 大小的确认
// 计算块的结尾及其大小LPCVOID pEnd = (PBYTE) pBlock + mbi.RegionSize;TCHAR szSize[MAX_PATH];:: StrFormatByteSize(mbi.RegionSize, szSize, MAX_PATH) ;
// 显示块地址和大小std :: cout.fill ('0') ;std :: cout<< std :: hex << std :: setw(8) << (DWORD64) pBlock<< "-"<< std :: hex << std :: setw(8) << (DWORD64) pEnd<< (:: strlen(szSize)==7? " (" : " (") << szSize<< ") " ;
// 显示块的状态switch(mbi.State) {case MEM_COMMIT :std :: cout << "Committed" ;break;case MEM_FREE :std :: cout << "Free" ;break;case MEM_RESERVE :std :: cout << "Reserved" ;break;}
// 显示保护if(mbi.Protect==0 && mbi.State!=MEM_FREE) {mbi.Protect=PAGE_READONLY;}ShowProtection(mbi.Protect);
// 显示类型switch(mbi.Type) {case MEM_IMAGE :std :: cout << ", Image" ;break;case MEM_MAPPED:std :: cout << ", Mapped";break;case MEM_PRIVATE :std :: cout << ", Private" ;break;}
// 检验可执行的影像TCHAR szFilename [MAX_PATH] ;if (:: GetModuleFileName ((HMODULE) pBlock, // 实际虚拟内存的模块句柄szFilename, //完全指定的文件名称MAX_PATH)>0) { //实际使用的缓冲区大小
// 除去路径并显示:: PathStripPath(szFilename) ;std :: cout << ", Module: " << szFilename;}std :: cout << std :: endl;
// 移动块指针以获得下一下个块pBlock = pEnd;}}
}
void ShowVirtualMemory() {
// 首先,让我们获得系统信息SYSTEM_INFO si;:: ZeroMemory(&si, sizeof(si) ) ;:: GetSystemInfo(&si) ;
// 使用外壳辅助程序对一些尺寸进行格式化TCHAR szPageSize[MAX_PATH];::StrFormatByteSize(si.dwPageSize, szPageSize, MAX_PATH) ;DWORD dwMemSize = (DWORD64)si.lpMaximumApplicationAddress -(DWORD64) si.lpMinimumApplicationAddress;TCHAR szMemSize [MAX_PATH] ;:: StrFormatByteSize(dwMemSize, szMemSize, MAX_PATH) ;
// 将内存信息显示出来std :: cout << "Virtual memory page size: " << szPageSize << std :: endl;std :: cout.fill ('0') ;std :: cout << "Minimum application address: 0x"<< std :: hex << std :: setw(8)<< (DWORD64) si.lpMinimumApplicationAddress<< std :: endl;std :: cout << "Maximum application address: 0x"<< std :: hex << std :: setw(8)<< (DWORD64) si.lpMaximumApplicationAddress<< std :: endl;std :: cout << "Total available virtual memory: "<< szMemSize << std :: endl ;
}
int main() {
//显示虚拟内存的基本信息ShowVirtualMemory();
// 遍历当前进程的虚拟内存::WalkVM(::GetCurrentProcess());return 0;
}

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

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

#LLMOps##AIGC# Dify_构建本地知识库问答应用-生成Al应用的创新引擎 用于构建助手API和GPT的开源开发平台

github&#xff1a; https://github.com/langgenius/dify/blob/main/README_CN.md 介绍文档&#xff1a;https://docs.dify.ai/getting-started/readme Dify 介绍 Dify 笔记 Dify 是什么&#xff1f; 开源的大语言模型&#xff08;LLM&#xff09;应用开发平台融合了后端即服…

“中国最危险的女人”闪亮现身瑞士

媒体报道&#xff0c;昨&#xff08;1月17日&#xff09;天中午&#xff0c;财新CEO午餐会在瑞士达沃斯举办&#xff0c;财新传媒社长胡舒立女士在致辞&#xff0c;她的现身&#xff0c;进一步粉碎了不久前“胡舒立被国安约谈”的网传谣言。 图&#xff1a;财新传媒社长胡舒立&…

Docker部署的gitlab升级指南(15.11.X容器里升级PostgreSQL到13.8)

一、确定当前版本 #进入当前版本容器产看gitlab版本 docker exec -it gitlab cat /opt/gitlab/embedded/service/gitlab-rails/VERSION#显示版本如下 14.4.0二、备份数据&#xff0c;防止升级发生意外 #执行备份命令 docker exec -ti gitlab gitlab-rake gitlab:backup:creat…

【K8S 云原生】K8S的对外服务—ingress

目录 一、K8S的Service 1、Service的作用 2、Service类型&#xff1a; 二、ingress 1、ingress的组成&#xff1a; 2、ingress资源的定义项&#xff1a; 3、ingress暴露服务端的方式 3.1、DeploymentLoadBalancer模式&#xff1a; 1、工作流程图&#xff1a; 3.2、Dae…

软件工程:黑盒测试等价分类法相关知识和多实例分析

目录 一、黑盒测试和等价分类法 1. 黑盒测试 2. 等价分类法 二、黑盒测试等价分类法实例分析 1. 工厂招工年龄测试 2. 规定电话号码测试 3. 八位微机测试 4. 三角形判断测试 一、黑盒测试和等价分类法 1. 黑盒测试 黑盒测试就是根据被测试程序功能来进行测试&#xf…

软件测试工程师应该怎么样?

目录 软件测试工程师的几种职位 测试经理 测试架构师 测试经理 VS 测试架构师 测试在管理上的发展 测试在技术上的发展 角色与段位 软件测试工程师的几种职位 测试经理 一个测试经理应该对业务有一定的了解&#xff0c;以便更好地管理测试团队和负责测试项目的工作…

网络编程【1】

【 1 】什么是网络编程 网络编程是指通过计算机网络进行数据交换和通信的编程过程。它涉及到使用网络协议和通信接口&#xff0c;使不同计算机之间能够进行数据传输和通信。 总结&#xff1a; 网络编程的研究前提就是基于互联网 网络编程就是基于互联网写代码 【 2 】为什么…

【印象深刻的实战经历】两次全国大学生数学建模经历分享

目录 &#x1f33c;初次接触 初次参加培训 分享培训所得 比赛开始 &#x1f525;再次接触 参加校赛 机缘巧合 再次培训 比赛开始 &#x1f4d5;技巧总结 从问题的实际意义分析大体上可分为 从问题的解决方法上分析 做国赛题目的步骤 赛前准备 选题 寻找思路…

给科研人的 ML 开源发布工具包

什么是开源发布工具包&#xff1f; 恭喜你的论文成功发表&#xff0c;这是一个巨大的成就&#xff01;你的研究成果将为学界做出贡献。 其实除了发表论文之外&#xff0c;你还可以通过发布研究的其他部分&#xff0c;如代码、数据集、模型等&#xff0c;来增加研究的可见度和采…

LeetCode三数之和

解题思路 常规玩法&#xff1a; 用三层循环暴力的把所有可能都试一遍&#xff0c;然后再去重。 进阶玩法&#xff1a;使用指针 排序去重遍历 就这三个步骤&#xff0c;每个步骤中细节挺多。 排序与去重原因&#xff1a; -2 -1 1 1 这是排序好的数据&#xff08;先不说…

预处理详解(#和##运算符、命名约定、#undef​​、命令行定义​、条件编译、头文件的包含​)

目录 一、#和## 1.1#运算符 1.2## 运算符​ 二、命名约定​ 三、#undef​ 四、命令行定义​ 五、条件编译​ 六、头文件的包含​ 4.1 头文件被包含的方式&#xff1a;​ 4.1.1 本地文件包含​ Linux环境的标准头文件的路径&#xff1a;​ VS环境的标准头文件的路径&…

学习记录-自动驾驶与机器人中的SLAM技术

以下所有内容均为高翔大神所注的《自动驾驶与机器人中的SLAM技术》中的内容 2D SLAM 作者实现了一个2D 的ICP 3D SLAM ICP 实现了一个并发的ICP配准实现了点到面的ICP实现了点到线的ICP点到线的ICP的结果与点到点的ICP相当&#xff0c;略差于点到面的、在三中算法中&#…