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

目录

前言

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.rhkb.cn/news/240641.html

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

相关文章

首届PolarDB开发者大会在京举办,阿里云李飞飞:云数据库加速迈向智能化

1月17日&#xff0c;阿里云PolarDB开发者大会在京举办&#xff0c;中国首款自研云原生数据库PolarDB发布“三层分离”新版本&#xff0c;基于智能决策实现查询性能10倍提升、节省50%成本。此外&#xff0c;阿里云全新推出数据库场景体验馆、训练营等系列新举措&#xff0c;广大…

QT第六天

要求&#xff1a;使用QT绘图&#xff0c;完成仪表盘绘制&#xff0c;如下图。 素材 运行效果&#xff1a; 代码&#xff1a; widget.h #ifndef WIDGET_H #define WIDGET_H#include <QWidget> #include <QPainter> #include <QPen>QT_BEGIN_NAMESPACE name…

S/MIME电子邮件证书申请指南

近年来&#xff0c;邮件安全问题日益突出&#xff0c;电子邮件成为诈骗、勒索软件攻击的重灾区。恶意邮件的占比屡创新高&#xff0c;邮件泄密事件更是比比皆是。在如此严峻的网络安全形势下&#xff0c;使用S/MIME电子邮件证书进行邮件收发是当今最佳的邮件安全解决方案之一。…

Apache Doris (六十四): Flink Doris Connector - (1)-源码编译

🏡 个人主页:IT贫道-CSDN博客 🚩 私聊博主:私聊博主加WX好友,获取更多资料哦~ 🔔 博主个人B栈地址:豹哥教你学编程的个人空间-豹哥教你学编程个人主页-哔哩哔哩视频 目录 1. Flink与Doris版本兼容

结构体内存对齐的跨平台做法

作者&#xff1a;朱金灿 来源&#xff1a;clever101的专栏 为什么大多数人学不会人工智能编程&#xff1f;>>> 之前写了一篇文章&#xff1a;使用标准C库读文件时需要注意的一个问题&#xff0c;今天发现是错误的。正确的做法是使用#pragma pack预处理指令。示例程序…

无需信用卡注册美区Apple ID指南

第一步 准备工作 1、一个没有注册过AppleID的邮箱&#xff0c;建议最好是Gmail邮箱 2、一个苹果手机&#xff0c;当然这个是必须的 3、需要科学上网 第二步 苹果网站注册 为了避免cookie的干扰&#xff0c;最好是在无痕模式下打开以上网页&#xff0c;创建你的AppleID&#…

【云原生】springboot 整合 OpenTelemetry

目录 一、前言 二、应用可观测性概述 2.1 什么是可观测性 2.2 可观测性三大指标 2.2.1 指标&#xff08;Metrics&#xff09; 2.2.2 日志&#xff08;log&#xff09; 2.2.3 追踪(Traces) 三、OpenTelemetry 介绍 3.1 什么是OpenTelemetry 3.2 OpenTelemetry架构和组件…

LiveGBS流媒体平台GB/T28181功能-基础配置接入控制白名单黑名单配置控制设备安全接入设备单独配置接入密码

LiveGBS基础配置接入控制白名单黑名单配置控制设备安全接入设备单独配置接入密码 1、白名单配置应用场景2、接入控制2.1、白名单2.2、黑名单 3、搭建GB28181视频直播平台 1、白名单配置应用场景 LiveGBS国标流媒体服务&#xff0c;支持白名单配置。 可在设备注册前&#xff0…

最新版git2.43安装、记住用户名和密码以及tortoisegit2.15使用

一、下载git 打开git官网地址&#xff1a;https://git-scm.com/进行下载 下载完安装&#xff0c;一直next就好&#xff0c;如果愿意就可以改下安装路径&#xff0c;改在d盘。 具体可以参考&#xff1a;git安装教程 二、安装完下载小乌龟以及中文语言包 下载地址&#xff1a;…

小程序中使用上传图片,显示、删除、预览

一、功能介绍 需要哦用户点击加号上传图片&#xff0c;并展示所上传图片和能够删除和预览 二、功能实现 采用的uniapp&#xff0c;创建了一个view容器包裹加号图标和展示的图片。 内部展示图片超过9张时候&#xff0c;加号图片隐藏 <view class"img-list">/…

12AOP面向切面编程/GoF之代理模式

先看一个例子&#xff1a; 声明一个接口&#xff1a; // - * / 运算的标准接口! public interface Calculator {int add(int i, int j);int sub(int i, int j);int mul(int i, int j);int div(int i, int j); }实现该接口&#xff1a; package com.sunsplanter.prox…

私有仓库工具Nexus Maven如何部署并实现远程访问管理界面

文章目录 1. Docker安装Nexus2. 本地访问Nexus3. Linux安装Cpolar4. 配置Nexus界面公网地址5. 远程访问 Nexus界面6. 固定Nexus公网地址7. 固定地址访问Nexus Nexus是一个仓库管理工具&#xff0c;用于管理和组织软件构建过程中的依赖项和构件。它与Maven密切相关&#xff0c;可…

谈谈前端开发中的防抖和节流

本文作者为 360 奇舞团前端开发工程师 李武阳 概述 防抖和节流是前端开发中常用的函数优化手段&#xff0c;它们可以限制函数的执行频率&#xff0c;提升性能和用户体验。主要用于处理高频触发的事件&#xff0c;例如&#xff1a;用户的滚动、输入、点击和表单的重复提交等。 防…

数据结构之list类

前言 list是列表类。从list 类开始&#xff0c;我们就要接触独属于 Python 的数据类型了。Python 简单、易用&#xff0c;很大一部分原因就是它对基础数据类型的设计各具特色又相辅相成。 话不多说&#xff0c;让我们开始学习第一个 Python 数据类型一list。 1. list的赋值 输…

[Android] Android架构体系(1)

文章目录 Android 的框架Dalvik 虚拟机JNI原生二进制可执行文件Android NDK中的binutils Bionic谷歌考虑到的版权问题Bionic与传统的C标准库&#xff08;如glibc&#xff09;的一些不同 参考 Android 的框架 Android 取得成功的关键因素之一就是它丰富的框架集。 没有这些框架…

【问题解决】cannot import name ‘circle‘ from ‘skimage.draw‘ 问题解决

使用 from skimage.draw import circle 报错如下&#xff1a; from skimage.draw import circle ImportError: cannot import name circle from skimage.draw (/home/onur/.local/lib/python3.9/site-packages/skimage/draw/__init__.py)修改&#xff1a; from skimage.draw …

Linux_Docker图形化工具Portainer如何安装并结合内网穿透实现远程访问

文章目录 前言1. 部署Portainer2. 本地访问Portainer3. Linux 安装cpolar4. 配置Portainer 公网访问地址5. 公网远程访问Portainer6. 固定Portainer公网地址 前言 本文主要介绍如何本地安装Portainer并结合内网穿透工具实现任意浏览器远程访问管理界面。Portainer 是一个轻量级…

2024年甘肃省职业院校技能大赛信息安全管理与评估 样题一 模块一

竞赛需要完成三个阶段的任务&#xff0c;分别完成三个模块&#xff0c;总分共计 1000分。三个模块内容和分值分别是&#xff1a; 1.第一阶段&#xff1a;模块一 网络平台搭建与设备安全防护&#xff08;180 分钟&#xff0c;300 分&#xff09;。 2.第二阶段&#xff1a;模块二…

时序预测 | MATLAB实现GRNN广义回归神经网络时间序列未来多步预测(程序含详细预测步骤)

时序预测 | MATLAB实现GRNN广义回归神经网络时间序列未来多步预测(程序含详细预测步骤) 目录 时序预测 | MATLAB实现GRNN广义回归神经网络时间序列未来多步预测(程序含详细预测步骤)预测效果基本介绍程序设计参考资料预测效果 基本介绍 MATLAB实现GRNN广义回归神经网络时间序列…