大页面的的应用场景及注意事项如下(出自北风网 VC++ 系列视频教程,请到官网下载尊重版权):

  • 大页面的内存是不能换页的,只能常驻于RAM中,不会存在页面文件中 (即大页面不会换页,物理内存紧张时慎用)
  • 大页面的内存算作进程的私有页面,不会统计在进程工作集中
  • 因为大页面尺寸远大于小页面尺寸,分配一个大页面必须要有连续的足够大块的物理内存,这可能导致系统进行耗时的内存碎片整理工作
  • 大页面始终是可读写的,并且是充分利用了高速缓存(TLB)
  • 大页面的分配不受作业对象对进程工作集大小的限制
  • 在Intel安腾架构上的WOW64系统不支持32位应用大页面,除非重编译应用为64位的

参考代码

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
#include <tchar.h>
#include <windows.h>
#include <time.h>

void Privilege(TCHAR* pszPrivilege, BOOL bEnable)
{
HANDLE hToken = NULL;
TOKEN_PRIVILEGES tp = {};
OpenProcessToken(GetCurrentProcess(), TOKEN_ADJUST_PRIVILEGES TOKEN_QUERY, &hToken);
LookupPrivilegeValue(NULL, pszPrivilege, &tp.Privileges[0].Luid);
tp.PrivilegeCount = 1;
tp.Privileges[0].Attributes = SE_PRIVILEGE_ENABLED;
AdjustTokenPrivileges(hToken, FALSE, &tp, 0, (PTOKEN_PRIVILEGES)NULL, 0);
CloseHandle(hToken);
}

int _tmain(int argc, _TCHAR* argv[])
{
// 设置随机数种子
srand((unsigned int)time(NULL));
// 提权
Privilege(TEXT("SeLockMemoryPrivilege"), TRUE);
// 获取大页面大小
SIZE_T szLargePage = GetLargePageMinimum();
// 保留空间 64 * szLargePage
PVOID pBuffer = VirtualAlloc(0, 64 * szLargePage, MEM_RESERVE, PAGE_READWRITE);
// 使用 MEM_LARGE_PAGES 标志提交内存
VirtualAlloc(pBuffer, 2 * szLargePage, MEM_COMMIT MEM_LARGE_PAGES, PAGE_READWRITE);
// 测试内存数据
double* pdbArray = (double*)pBuffer;
for (int i = 0; i < (int)(2 * szLargePage / sizeof(double)); i++)
{
pdbArray[i] = 1.0f * rand();
}
// 释放内存
VirtualFree(pBuffer, 0, MEM_RELEASE);
return 0;
}