前言
课程设计开始了,实验很有意思,写博客总结学到的知识
白嫖容易,创作不易,学到东西才是真
本文原创,创作不易,转载请注明!!!
本文链接
个人博客:https://ronglin.fun/archives/179
PDF链接:见博客网站
CSDN: https://blog.csdn.net/RongLin02/article/details/118308841
为了美观,实验源代码在结尾处,整合版见下
链接:https://pan.baidu.com/s/1rXj1QJGuw-BVc5sQWret9w
提取码:Lin2
操作系统课程设计源代码
本次操作系统课程设计合集
操作系统课设之Windows 进程管理
操作系统课设之Linux 进程管理
操作系统课设之Linux 进程间通信
操作系统课设之Windows 的互斥与同步
操作系统课设之内存管理
操作系统课设之虚拟内存页面置换算法的模拟与实现
操作系统课设之基于信号量机制的并发程序设计
操作系统课设之简单 shell 命令行解释器的设计与实现
仅用于学习,如有侵权,请联系我删除
实验题目
内存管理
实验目的
了解 Windows 的内存结构和虚拟内存的管理,理解进程的虚拟内存空间和物理内存的映射关
系。加深对操作系统内存管理、虚拟存储管理等理论知识的理解。
实验内容
实验步骤:
在Windows下用codeblocks创建一个project,将指导书的源码copy到main.cpp下,编译运行,查看结果。
报错1:StrFormatByteSize()
不可用
因为没链接库
在左侧点击工程文件夹 – 右键 – bulid options…,
在弹出的窗口选择–linker settings – other linker options中
添加-lshlwapi
程序正常运行。
有同学还遇到了类型错误的报错,(相对于codeblocks20.03版本,本人是17.12)
需要把报错位置的(DWORD)
改成(DWORD64)
即可
结果如下
虚拟内存每页容量为:4.00 KB__________
最小应用地址:0x00010000__________
最大应用地址:______0x7ffeffff ___________
当前可供应用程序使用的内存空间为:___1.99 GB _____
当前计算机的实际内存大小为:16G
理论上每个 Windows 应用程序可以独占的最大存储空间是:17.99G
实验结果与分析
因为程序输出是格式化输出,所以写了一个C++程序,用来格式化处理数据,格式化处理有删除,
字符,然后在按照虚存地址类型排序输出。
整理好的结果如下(部分)
由于实在太多,截图不方便,这里只列出部分。
Windows Xp 是 32 位的操作系统,它使计算机 CPU 可以用 32 位地址对 32 位内存块进行操作。
内存中的每一个字节都可以用一个 32 位的指针来寻址。这样,最大的存储空间就是2^32字节或 4000兆字节 (4GB) 。这样,在 Windows 下运行的每一个应用程序最大可能占有 4GB 大小的空间。然而,实际上每个进程一般不会占有 4GB 内存。Windows 在幕后将虚拟内存 (virtual memory,VM) 地址映射到了各进程的物理内存地址上。而所谓物理内存是指计算机的 RAM 和由 Windows分配到用户驱动器根目录上的换页文件。物理内存完全由系统管理。在 Windows 环境下,4GB 的虚拟地址空间被划分成两个部分:低端 2GB 提供给进程使用,高端 2GB 提供给系统使用。这意味着用户的应用程序代码,包括 DLL 以及进程使用的各种数据等,都装在用户进程地址空间内 (低端 2GB) 。用户进程的虚拟地址空间也被分成三部分:
- 虚拟内存的已调配区 (committed) :具有备用的物理内存,根据该区域设定的访问权限,用户可以进行写、读或在其中执行程序等操作。
- 虚拟内存的保留区 (reserved) :没有备用的物理内存,但有一定的访问权限。
- 虚拟内存的自由区 (free) :不限定其用途,有相应的 PAGE_NOACCESS 权限。
与虚拟内存区相关的访问权限告知系统进程可在内存中进行何种类型的操作。例如,用户不能在只有 PAGE_READONLY 权限的区域上进行写操作或执行程序;也不能在只有 PAGE_EXECUTE权限的区域里进行读、写操作。而具有 PAGE_ NOACCESS 权限的特殊区域,则意味着不允许进程
对其地址进行任何操作。在进程装入之前,整个虚拟内存的地址空间都被设置为只有 PAGE_NOACCESS 权限的自由区域。当系统装入进程代码和数据后,才将内存地址的空间标记为已调配区或保留区,并将诸如EXECUTE、READWRITE 和 READONLY 的权限与这些区域相关联。
小结与心得体会
对于Windows我们每天都在使用,但是对于其的内存管理倒是没怎么关注过,本次实验了解了Windows的内存管理机制。
源代码
// 工程 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 ;
}
int main()
{
//显示虚拟内存的基本信息
ShowVirtualMemory();
// 遍历当前进程的虚拟内存
::WalkVM(::GetCurrentProcess());
return 0;
}