0

我正在为一个统计包编写一些代码,我首先将数据读入一个指针数组。我使用 malloc 初始化指针并分配足够的内存;但是,我有时会在下面的代码末尾遇到内存分配错误。

#include <stddef.h>
#include <stdio.h>
#include <stdlib.h>
#include "stats.h"

int main(int argc, char **argv) {
FILE *fp, *outFile, *outBin;    // create a file identifier
size_t n, nbins; // # of data points
double *data; // pointer to hold data
double average, variance, *med; // stat returns
int medianComplete, histComplete, i; // return 1 on success
hist_t *Histogram;

// read in the number of bins from exe arguments
nbins = atoi(argv[1]);
nbins = (size_t)nbins;

// open the binary datafile and read in first value
// which is number of data points
// use exe input for filename
fp = fopen(argv[2],"rb");
fread(&n, sizeof(size_t),1,fp);

// allocate enough memory to hold all data
data = (double*)malloc(sizeof(double)*n);
if (!data)
    printf("Memory allocation error");
fread(data,sizeof(double),n,fp);

该程序在我的个人计算机 (MacOSX) 上编译并运行良好,但在我尝试在 Linux 服务器上运行时由于分段错误而失败。我使用 Valgrind 来查看是否可以追踪错误并收到以下结果。

==8641== Warning: silly arg (-501426814648844128) to malloc()
==8641== Invalid write of size 1
==8641==    at 0x4C2B20D: mempcpy (mc_replace_strmem.c:956)
==8641==    by 0x4EA2F15: _IO_file_xsgetn (fileops.c:1423)
==8641==    by 0x4E971D2: fread (iofread.c:44)
==8641==    by 0x40086D: main (runstats.c:28)
==8641==  Address 0x0 is not stack'd, malloc'd or (recently) free'd

这是我编写的第一个使用指针的程序,我不知道为什么它可以在一个系统上运行,而在另一个系统上不行。

4

2 回答 2

5

sizeof(size_t)是平台相关的。在具有默认 32 位代码的 OSX 上,size_t是 4 个字节。在 64 位 Linux 上,size_t是 8 个字节。如果您在 64 位 Linux 上运行,那么您阅读的内容与n在 32 位 OSX 上阅读的内容不同。

如果必须使用二进制数据格式,请不要使用平台特定类型的大小作为字段大小。确定文件头是 4 字节还是 8 字节,小端还是大端,并始终使用它。

请参阅: 32 位与各种 64 位数据模型的 sizeof(size_t) 是什么?

于 2012-02-03T08:32:44.970 回答
2

您绝对应该在发布之前格式化您的来源......

执行后使用调试器或 printf n 的值fread(&n, sizeof(size_t),1,fp);。似乎你没有得到你期望的价值。

于 2012-02-03T08:07:40.757 回答