0

我在具有 18 GB RAM 的 64 位 Ubuntu 机器上运行以下代码,如您所见,当我尝试分配 2^31 字节时,我对 Malloc 的调用失败。我不确定为什么会发生这种情况,或者如何解决它(我已经尝试过编译器标志和 calloc())。我想知道是否有人可以向我解释为什么我无法在 64 位机器上分配更多空间以及如何解决此问题。

#include <stdio.h>
#include <stdlib.h>
//#include "svm_model_matlab.h"
//include "svm.h"
#include <math.h>


struct svm_node
{
        int index;
        double value;
};


//#define Malloc(type,n) (type *)calloc(n,sizeof(type))
#define Malloc(type,n) (type *)malloc((n)*sizeof(type))

int main()
{

        int i;
        for(i =25; i< 35; ++i)
        {
                printf("2^%d %d \n", i,(long int) pow(2,i));
                svm_node* x_space = Malloc(struct svm_node,pow(2,i));
                printf("the address is %x\n", x_space);
                free(x_space);
        }


        return 0;
}

输出:

2^25 33554432
the address is 8513e010
2^26 67108864
the address is 6513e010
2^27 134217728
the address is 2513e010
2^28 268435456
the address is a513e010
2^29 536870912
the address is a513e010
2^30 1073741824
the address is 0
2^31 -2147483648
the address is 0
2^32 0
the address is 0
2^33 0
the address is 0
2^34 0
the address is 0

更新:

我发现了我遇到的问题:我目前在 64 位 Ubuntu linux 发行版上的 EC2 上运行我的代码,而 EC2 上的默认 linux 框有 0 个交换空间。这导致我的进程在请求比物理 RAM 更多的内存时出现段错误,因为它无法分页。创建交换文件后,我的问题就消失了。

谢谢你的帮助

4

1 回答 1

4

pow()是计算 2 的幂的一种可怕方法。1 << i改为使用。

然后,选择一个足以容纳您请求的大小的数据类型。现在它溢出了 的大小int,因此试图分配负数的字节。由于明显的原因,这不起作用。

我怀疑这malloc(1ULL << 31)会在您的系统上成功,没有任何问题。

接下来,您分配的远远超过您的问题提到的 2 31个字节,您实际上是在尝试分配 2 i *sizeof (svm_node)或大约 2 i+4。失败的分配,i=30,大约是 16GB,这很可能比你的可用 RAM 还多。

最后,在打印指针时,您将获得 32 位值。试试printf("%p", x_space);吧。如果这仍然为您提供 32 位值,请尝试使用 64 位编译器。

于 2011-02-16T03:51:08.183 回答