【发布时间】:2014-01-31 15:08:45
【问题描述】:
我有一个文件有num 行:每一行都包含一个数字。我想将每个数字保存到向量*vet 中。这两个版本哪个更好?
[版本 1] 我有两个功能:第一个用于计算 num,第二个用于将数字保存到 *vet。我在main() 中用malloc 分配内存。
#include <stdio.h>
#include <stdlib.h>
/* The first function counts lines number */
int count_line (int *num)
{
FILE *fin;
char buff[10];
*num = 0;
if ( !(fin = fopen("numbers.dat", "r")) )
return 1;
while ( fgets(buff, sizeof(buff), fin) )
(*num)++;
return fclose(fin);
}
/* The second function save numbers into a vector */
int save_numbers (int *vet)
{
FILE *fin;
int i=0;
char buff[10];
if ( !(fin = fopen("numbers.dat", "r")) )
return 1;
while ( fgets(buff, sizeof(buff), fin) )
{
sscanf (buff, "%d", &vet[i]);
i++;
}
return fclose(fin);
}
int main ()
{
int num, i, *vet;
if ( count_line(&num) )
{
perror("numbers.dat");
exit(1);
}
vet = (int *) malloc ( num * sizeof(int) );
if ( save_numbers(vet) )
{
perror("numbers.dat");
exit(2);
}
/* print test */
for (i=0; i<num; i++)
printf ("%d ", vet[i]);
printf("\n");
free(vet);
return 0;
}
[版本 2] 我只有一个功能:它使用realloc 分配内存并将数字保存到*vet。
#include <stdio.h>
#include <stdlib.h>
/* This function allocate memory
and save numbers into a vector */
int save_numbers (int **vet, int *num)
{
FILE *fin;
int i = 0;
char buff[10];
if ( !(fin = fopen("numbers.dat", "r")) )
return 1;
while ( fgets(buff, sizeof(buff), fin) )
{
*vet = (int *) realloc (*vet, (i+1) * sizeof(int) );
sscanf (buff, "%d", &(*vet)[i]);
i++;
}
*num = i;
return fclose(fin);
}
int main ()
{
int i, num, *vet = NULL;
if ( save_numbers(&vet, &num) )
{
perror("numbers.dat");
exit(1);
}
/* print test */
for (i=0; i<num; i++)
printf ("%d ", vet[i]);
printf("\n");
free(vet);
return 0;
}
此处的文件示例:http://pastebin.com/uCa708L0
【问题讨论】:
-
最好尽量少读磁盘。所以,第 2 版。
-
正如@APerson 所说,磁盘 I/O 很昂贵,所以版本 2 更好。但这并不好;一般来说,增量内存分配的成本是二次方的。您应该计划在每次分配时将分配的空间量加倍,以摊销分配的成本。
-
并避免在为数字 N 分配空间时将前 N-1 个数字从一个地方复制到另一个地方的成本。这并不总是发生,但正式地,
realloc()释放了它当前分配的空间并分配新空间(但有时新旧指针会相同)。 -
@Jonathan Leffler 那么如果我能解决你所说的问题我应该怎么做?
标签: c file function malloc realloc