Win API 内存整理

时间:2024-12-23 21:35:26

记得我的笔记本上曾经安装了一款名为内存整理大师的软件,当时觉得挺好用而且挺NB的,就是导致开机启动有点慢。

当时我就在想,内存整理是怎么实现的?不过那是水平实在是不怎么样,估计连windows程序的消息概念都不清楚吧。所以虽然不明白原因,但是觉得,很NB。

今天看到网上有关于EmptyWorkingSet函数的介绍,然后就看了下。看那几个函数的调用,不是很明白,所以直接去MSDN查看函数原型以及介绍。

看明白了那两个函数,就突然觉得,实现内存整理太简单了……

首先摆出来几个API函数,分别是:

EnumProcess http://msdn.microsoft.com/en-us/library/windows/desktop/ms682629(v=vs.85).aspx

OpenProcess http://msdn.microsoft.com/en-us/library/windows/desktop/ms684320(v=vs.85).aspx

SetProcessWorkingSetSize http://msdn.microsoft.com/en-us/library/windows/desktop/ms686234(v=vs.85).aspx

EmptyWorkingSet http://msdn.microsoft.com/en-us/library/windows/desktop/ms682606(v=vs.85).aspx

看原型,点连接……

其中第一个函数的作用,就是枚举全部的运行中进程,将其所有的进程ID放入到一个数组中。

第二个函数的作用,是根据进城ID获取到进程句柄。这里需要提一点的就是,在其flag标志位中需要特别设置

第三个函数的作用,设置此进程的工作集大小。

第四个函数的作用,清理指定进程工作集中未使用的页。

这里涉及一个概念,就是工作集。什么是工作集呢?其实最简单的理解,就是进程一映射到物理内存的部分。

这个概念的出现,是因为如果把一个进程的全部代码和资源都映射到内存中,必然存在很大的浪费,毕竟一个程序的执行代码使用率不是均等的。把使用率高的代码和资源映射到内存,把使用率低的代码调到虚拟从内存中显然是很明智的选择。我就理解到这啦,如果理解有误,欢迎更正。

所以步骤如下:

1调用 EnumProcess 函数获取进程ID列表

2做一个循环,对其进行如下访问:

2.1调用OpenProcess函数以PROCESS_SET_QUOTA权限打开进程句柄。

2.2调用SetProcessWorkingSetSize函数来对指定进程的工作集进行紧缩,后两个参数只均为-1,尽可能空余出来未使用的页(page)。

2.3调用EmptyWorkingSet函数来清除进程中未使用的页

2.4关闭句柄

3 完成

简单的很是吧?没错,就是这么简单。

所以现在对于当时的那个什么内存管理大师特别不屑一顾,还好意思说大师……那可真是遍地都是大师了。

运行效果如下:

整理前:Win API 内存整理

整理后:

Win API 内存整理

简单整理一下代码,贴下来如下:

 #include <windows.h>
#include <iostream>
#include <cstdlib>
#include <Psapi.h>
#include <TlHelp32.h>
#include <assert.h>
#pragma comment(lib, "Iphlpapi")
#pragma comment(lib, "Psapi") int _tmain(int argc, _TCHAR* argv[])
{
DWORD pdwPIDArray[];
TCHAR StrBuf[];
DWORD dwPIDArraySize;
EnumProcesses(pdwPIDArray, , &dwPIDArraySize);
for(int i = ; i < dwPIDArraySize; ++ i)
{
HANDLE hProcess = OpenProcess(PROCESS_QUERY_INFORMATION|PROCESS_VM_READ|PROCESS_SET_QUOTA, , pdwPIDArray[i]);
SetProcessWorkingSetSize(hProcess, -, -);
EmptyWorkingSet(hProcess);
CloseHandle(hProcess);
}
system("pause");
return ;
}