gpt4 book ai didi

c - 从二进制文件读取/写入链接节点

转载 作者:太空宇宙 更新时间:2023-11-04 08:21:27 24 4
gpt4 key购买 nike

我在读取包含链接节点的二进制文件时遇到问题。

这是代码:

lib1.c

struct my_stack_node {
void *data;
struct my_stack_node *next;
};

struct my_stack {
int size;
struct my_stack_node *first;
};

int my_stack_write(struct my_stack *stack, char *filename){
int count = 0;
struct my_stack_node *aux;
FILE *file = fopen(filename, "wb");
if(stack->first != NULL){
aux = stack->first;
count++;
while(aux->next != NULL){
fwrite(&aux ,sizeof(aux), 1, file);
aux = aux->next;
count++;
}
}
fwrite(&stack, sizeof(stack), 1, file); //Escriure stack
fclose(file);
return count;
}

struct my_stack *my_stack_read(char *filename){
struct my_stack *stackRead;
struct my_stack_node *stackNode;
FILE *file = fopen(filename, "rb");

if(!file){
puts("Impossible obrir el fitxer");
return NULL;
}else{
int primerInici = 0;

while(!feof(file)){
if(primerInici == 0){
stackRead = (struct my_stack*) malloc(sizeof(struct my_stack));
fread(stackRead, sizeof(stackRead), 1, file);
primerInici = 1;
}else{
//Crear nou node i llegir-lo del fitxer
stackNode = (struct my_stack_node*) malloc(sizeof(struct my_stack_node));
fread(stackNode, sizeof(stackNode), 1, file);
//Afegir node a la pila
stackNode->next = stackRead->first;
stackRead->first = stackNode;
}
}
fclose(file);
return stackRead;
}

}

主.c

 struct my_data {
int val;
char name[60];
};


int main() {
struct my_stack *s, *t, *u;
struct my_data *data, *data1, *data2;
//...more code
u = my_stack_read("/tmp/my_stack.data");
if (! u) {
puts("Error in my_stack_read (u)");
exit(1);
}

if (my_stack_len(s) != my_stack_len(u)) {
puts("Stacks s and u don't have the same len");
exit(1);
}

// Test we can free the data and compare stacks s and u
while((data1 = my_stack_pop(s))) {
data2 = my_stack_pop(u);
if (! data2 || data1->val != data2->val || my_strcmp(data1->name, data2->name)) {
printf("Data in s and u are not the same: %d <> %d\n", data1->val, data2->val);
exit(1);
}
free(data1);
free(data2);
}
//...more code
puts("All tests passed");
return 0;

}

执行结果为:

堆栈长度:100

s和u中的数据不一样:22145808 <> 22134800

正确的结果应该是:

所有测试都通过了

最佳答案

这就是问题所在(在 my_stack_write 内部):

aux = stack->first;
count++;
while(aux->next != NULL){
fwrite(&aux ,sizeof(aux), 1, file);
aux = aux->next;
count++;
}

你正在写指针aux。不是 aux 指向的结构。也不是data指向的数据,这是重要的部分。

所以。想象一下你有这样的东西:

my_stack  { first=0x100 }
at memoryPosition 0x100 we have : my_stack_node { data=0x200; next=0x300 }
at memoryPosition 0x300 we have : my_stack_node { data=0x500; next=0x600 }
at memoryPosition 0x600 we have : my_stack_node { data=0x700; next=NULL }

对于您的程序正在编写的结构:0x100、0x300
您正在编写构成链表的节点的内存地址。而且你错过了最后一个节点,这是一种不同类型的错误。
但那是没有用的。下次您运行程序时,您的节点可能位于不同的内存地址中,因此没有必要保存它们。它是动态内存,每次运行程序时它可能驻留在不同的地方。

你应该写的是你的链接列表列出的数据。

几乎整个程序都重复了同样的错误。

如何正确写入链表中包含的数据:

void writeStack(struct my_stack *stack, const char *filename)
{
struct my_stack_node *aux;
FILE *file = fopen(filename, "wb");
if ( file==NULL )
{
fprintf( stderr, "Could not open %s for writting.\n", filename );
exit(1);
}
if (stack != NULL)
{
aux = stack->first;
while(aux != NULL)
{
// aux->data is of type void*
// Assuming that aux->data contains a struct my_data
// Most likely it would be better to redefine data as having
// type struct my_data*
fwrite(aux->data ,sizeof(struct my_data), 1, file);
aux = aux->next;
}
}
fclose(file);
}

这里我们遍历列表中的所有节点。
对于每一个,我们都写入其中包含的数据。
注意fwrite( aux->data, 是如何写入aux->data 指向的数据的,这是正确的。
fwrite( &aux, 会写入包含在 aux 中的内存地址,这不太可能是正确的。
fwrite( &aux->data, 会写入包含在 aux->data 中的内存地址,这也不太可能是正确的。

由您来添加计数代码和编写读取函数。

关于c - 从二进制文件读取/写入链接节点,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/33241238/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com