Array of structs replacing values over itself

后端 未结 3 1337
有刺的猬
有刺的猬 2020-11-30 15:25

Ok so I have the below code and I am just pulling various things from a file and inputing them in an array of structs, it \"seemingly\" works initially, BUT when I go to pri

相关标签:
3条回答
  • 2020-11-30 16:09

    This is a simple misunderstanding of pointers and char arrays (strings). Here are a couple pages that explains them pretty well:

    • http://www.cplusplus.com/doc/tutorial/pointers/
    • http://www.cplusplus.com/doc/tutorial/ntcs/

    In your case, you are setting your struct pointer values equal to the returned pointer from strtok. A lot of those string functions work by putting the result at a certain memory address and returning the pointer to it. The pointer returned is always the same, so all your struct values are going to point to the last result of the strtok call.

    This is why you need strdup (String duplicate). Basically it takes the value at the address given and copies the contents into a new place in memory and returns the value.

    0 讨论(0)
  • 2020-11-30 16:10

    The error is here.

    d[counter].name=name;
    

    replace with:

    d[counter].name = strdup(name); /*don't forget to free this memory.*/
    

    the issue for the courses is the same.

    0 讨论(0)
  • 2020-11-30 16:18

    The strtok returns a pointer to the buffer and does not allocate memory. Since you do not copy the strings, you end up with lots of pointers pointing to the same buffer that is overwritten at each iteration of the loop.

    To fix this, you need to change your loop to copy the strings using strdup:

    while(fgets(buffer, sizeof(buffer), fp) != NULL)
    {
        d[counter].name = strdup(strtok(buffer, del));
        d[counter].course = strdup(strtok(NULL, del));
        d[counter].grade = atoi(strtok(NULL, del));
        counter++;
    }
    

    Don't forget to return the allocated memory with free once you no longer need the strings:

    for (i = 0; i < counter; i++) {
       free(d[i].name);
       free(d[i].course);
    
       d[i].name = NULL;
       d[i].course = NULL;
    }
    

    Note that strdup is part of POSIX1.2001 standard, not part of C89. If it is not available, you'll have to re-implement it yourself (quite easy):

    char *my_strdup(const char *str) {
      char *copy;
      size_t len = strlen(str) + 1;
      if (len == 0) return NULL;
      copy = (char *)malloc(len);
      if (copy == NULL) return NULL;
      memcpy(copy, str, len);
      return copy;
    }
    
    0 讨论(0)
提交回复
热议问题