4

可能重复:
您可以在 c 或 c++ 中分配一个非常大的单个内存块(> 4GB)吗?

我在我的电脑上运行以下程序:

#include <stdio.h>
#include <stdlib.h>

#define ONE_GIGABYTE 1024*1024*1024


int main(void) {
    int ctr=0;

    for (;;) {
        char *ptr = (char*)malloc(ONE_GIGABYTE*sizeof(char));
        if (ptr == 0)
            return -1;

        ctr++;
        printf("%d\n", ctr);
    }
}

flyrev@stargazer:~/weirdstuff$ ulimit -a
core file size          (blocks, -c) 0
data seg size           (kbytes, -d) unlimited
scheduling priority             (-e) 0
file size               (blocks, -f) unlimited
pending signals                 (-i) 128957
max locked memory       (kbytes, -l) 64
max memory size         (kbytes, -m) unlimited
open files                      (-n) 1024
pipe size            (512 bytes, -p) 8
POSIX message queues     (bytes, -q) 819200
real-time priority              (-r) 0
stack size              (kbytes, -s) 8192
cpu time               (seconds, -t) unlimited
max user processes              (-u) 128957
virtual memory          (kbytes, -v) unlimited
file locks                      (-x) unlimited
flyrev@stargazer:~/weirdstuff$ free -g
             total       used       free     shared    buffers     cached
Mem:            15          6          8          0          0          4
-/+ buffers/cache:          2         13
Swap:            9          0          9
flyrev@stargazer:~/weirdstuff$ clang malloc-program.c 
flyrev@stargazer:~/weirdstuff$ ./a.out 
1
2
flyrev@stargazer:~/weirdstuff$ 

这里发生了什么?

4

2 回答 2

4

您没有用完内存,而是在 32 位系统上运行,因此地址空间不足,您可能有理由期望能够在 32 位系统上分配 4Gb,因为:

2^32 = 4Gb

但是,在大多数操作系统上,至少 50% 的可用地址空间实际上是为内核保留的,因此您只能拥有其中的一半。

在 Linux 上,通过切换到使用PAE 内核,可以在 32 位模式下使用显着超过 4Gb 的数据。如果您需要,许多 Linux 发行版都将 PAE 内核作为一个包提供。

编辑:正如 Dietrich 所指出的:PAE 允许使用更多内存,但仍然只为您提供 4 GiB 的地址空间。因此,使用 16 GiB,您可以拥有 8 个程序,每个程序 2 GiB,但您仍然不能拥有超过 2-ish GiB 的程序

于 2012-10-20T15:00:44.690 回答
1

@Benj 为这个问题提供了一个很好的答案。但是,我想补充一点,即使一个人使用的是 64 位系统,如果使用针对 32 位系统的编译器编译程序,他可能仍然只能使用 2GB 内存。

64 位系统通常支持 32 位程序。但是 32 位系统无法运行 64 位系统。因此,如果作者选择了不期望占用 2GB 内存的程序,可能会编译为针对 32 位系统。此外,32 位目标通常是默认值,即使在 64 位系统上也是如此。

于 2012-10-20T15:29:37.357 回答