31

我是开发 Delphi 应用程序的团队的成员。内存需求很大。500 MB 是正常的,但在某些情况下会出现内存不足异常。在这种情况下分配的内存通常在 1000 - 1700 MB 之间。

我们当然想要 64 位编译器,但现在不会发生(如果发生,我们还必须转换为 unicode,但那是另一回事了……)。

我的问题是为什么在 64 位环境中运行时每个进程有 2 GB 的内存限制。指针是 32 位的,所以我认为 4 GB 是正确的限制。我使用德尔福 2007。

编辑: 因此,如果我在 Delphi 中使用以下方法设置 IMAGE_FILE_LARGE_ADDRESS_AWARE 标志:

{$SetPeFlags IMAGE_FILE_LARGE_ADDRESS_AWARE}

在 Windows Server 2003 x64 上运行生成的 Exe 文件,那么应用程序可以寻址 4 GB 吗?

  • 我应该在 boot.ini 中设置 /3GB 开关吗?
  • 我们已经尝试过,但在 32 位 Windows Server 2003 上,它似乎限制了 Windows 资源。日志中出现了更多带有 GDIError 的“内存不足”异常。但是在 64 位操作系统中运行时,这可能会消失吗?
4

4 回答 4

32

如果您使用 /LARGEADDRESSAWARE 标志编译 Delphi 应用程序,它将能够在 64 位操作系统上处理全部 4GB。否则,当在 WOW32 中运行时,操作系统假定应用程序期望的环境与它在 32 位操作系统上的环境相同,这意味着在 4GB 的地址空间中,2GB 专用于操作系统,2GB 分配给应用程序。

于 2010-04-29T19:51:23.467 回答
14

Delphi 中在 PE 可执行文件中设置 LARGEADDRESSAWARE 标志的语法是:

{$SetPEFlags IMAGE_FILE_LARGE_ADDRESS_AWARE}

把它放在你的 .dpr 文件中。

于 2010-04-29T21:13:30.633 回答
6

http://msdn.microsoft.com/en-us/library/aa366778(VS.85).aspx

每个 32 位进程的用户模式虚拟地址空间:2 GB

于 2010-04-29T19:46:56.780 回答
2

只要您选择接收设置了高位的 32 位指针(通过包含LARGE_ADDRESS_AWAREPE 标志),就没有 2GB 的限制。

直接观察

var
   p: Pointer;
   n: Int64;
begin
   p := Pointer($D0000000); //Above the 2GB line; and the 3GB line!

   p := VirtualAlloc(p, 1024, MEM_COMMIT or MEM_RESERVE, PAGE_READWRITE);
   if p = nil then
      RaiseLastWin32Error;

   n := Cardinal(p);
   ShowMessage(IntToHex(n, 16));
end;

在此处输入图像描述

结论

在 64 位 Windows 上没有 2GB 的限制,只要你发誓你可以处理高于 $7FFFFFFF 的指针。

注意:任何代码都会发布到公共领域。无需归属。

于 2013-09-30T20:26:07.040 回答