操作系统课设——内存管理

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

二、 实验内容和步骤
了解和检测进程的虚拟内存空间。
步骤 1:创建一个“Win32 Consol Application”工程,然后拷贝清单 5-1 中的程序,编译成可执行文件。
步骤 2:在 VC 的工具栏单击“Execute Program”(执行程序) 按钮,或者按 Ctrl + F5 键,或者在“命
令提示符”窗口运行步骤 1 中生成的可执行文件。
步骤 3:根据运行结果,回答下列问题
虚拟内存每页容量为:4.00kb________
最小应用地址:0x00010000____
最大应用地址:0x7ffeffff
当前可供应用程序使用的内存空间为:1.99GB
___
当前计算机的实际内存大小为:cmd进入后输入命令systeminfo可知其物理内存总量
理论上每个 Windows 应用程序可以独占的最大存储空间是:4GB_
提示:可供应用程序使用的内存空间实际上已经减去了开头与结尾两个 64KB 的保护区。
虚拟内存空间中的 64KB 保护区是防止编程错误的一种 Windows 方式。任何对内存中这一区域
的访问 (读、写、执行) 都将引发一个错误陷阱,从而导致错误并终止程序的执行。
按 committed、reserved、free 等三种虚拟地址空间分别记录实验数据。其中“描述”是指对该组
数据的简单描述,例如,对下列一组数据:
00010000 – 00012000 <8.00KB> Committed, READWRITE, Private
可描述为:具有 READWRITE 权限的已调配私有内存区。

三、参考程序

// 工程 vmwalker
#include <windows.h>
#include <iostream>
#include <shlwapi.h>
#include <iomanip>
#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) << (DWORD) pBlock
 << "-"
 << std :: hex << std :: setw(8) << (DWORD) 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 = (DWORD)si.lpMaximumApplicationAddress -
 (DWORD) 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)
 << (DWORD) si.lpMinimumApplicationAddress
 << std :: endl;
 std :: cout << "Maximum application address: 0x"
 << std :: hex << std :: setw(8)
 << (DWORD) si.lpMaximumApplicationAddress
 << std :: endl;
 std :: cout << "Total available virtual memory: "
 << szMemSize << std :: endl ;
}
void main()
{
//显示虚拟内存的基本信息
ShowVirtualMemory();
 // 遍历当前进程的虚拟内存
::WalkVM(::GetCurrentProcess());
}

四、实验结果以及分析
在这里插入图片描述
结果分析:该程序从主函数Main()出发,调用void WalkVM(HANDLE hProcess)函数,void WalkVM(HANDLEhProcess)函数获得系统信息,分配应用程序内存地址空间。然后开始做循环,从函数运行开始每次获得下一个虚拟程序内存块的信息,之后计算块的结尾及大小,然后再显示块的大小与位置,状态,显示保护方式(void ShowProtection(DWORD dwTarget)),将文件名显示出来,移动块指针获得下一块,依次这样循环下去,直至结束。
五、小结与心得体会
通过本实验,我明白了虚拟内存机制对于 Windows 操作系统的重要性,Windows 必须借助它的虚拟内存来维持系统和进程的运行。对页式存储管理有了进一步的了解,理解了Virtual QueryEX() API 等虚拟内存API的作用以及使用方法。

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值