“Pointer being freed was not allocated.” error aft

2020-04-05 07:35发布

问题:

I have this error with the following code:

int main(){
    point   *points = malloc(sizeof(point));
    if (points == NULL){
        printf("Memory allocation failed.\n");
        return 1;
    }
    other_stuff(points);
    free(points);
    return 0;
}
void other_stuff(point points[]){
    //stuff
    realloc(points, number*sizeof(point))
}

I have searched, but found only examples where it was clear there was no allocation.

Here, I used malloc to initialise points, and later changed its size with realloc; so how is the pointer "not allocated" when I come to free it?

回答1:

realloc may move the memory to a new location (if there is not enough space to expand the old pointer). If that happens, you need to free the new pointer.

Try this tweak:

int main(){
    point   *points = malloc(sizeof(point));
    if (points == NULL){
        printf("Memory allocation failed.\n");
        return 1;
    }
    other_stuff(&points);
    free(points);
    return 0;
}
void other_stuff(point **points){
    //stuff
    point *temp = realloc(*points, number*sizeof(point));
    if(temp != NULL) {
      *points = temp;
      // and do your stuff
    }
    else {
      // panic? memory reallocation failed. Deal with it gracefully.
    }
}

By passing a handle to other_stuff, we give it control not only over the place where the pointer is pointing, but to the address of the pointer itself. This allows it to move the memory around. Handles are a good way to manage memory dynamically; but conceptually a pointer to a pointer takes some getting used to...



回答2:

realloc returns a new pointer. That is what you need to free (eventually) if the function succeeded. Otherwise, it failed, and you keep the old pointer around for this case.

How to use realloc:

whataver *f = malloc(count * sizeof(*f));
/* ... */
whatever *temp = realloc(f, new_count * sizeof(*temp));
if (temp)
    f = temp;  // realloc worked, f is no longer valid/needed
else
    free(f);   // error

realloc may return the same pointer, or it may not. The point is that you no longer care about the original pointer if realloc succeeded. If it had to allocate a new block then the original pointer is invalid. If it didn't then it returned the same pointer, and you certainly don't want to deallocate that right away.



回答3:

Thank you for the solution : realloc (MAY) return a NEW pointer.

Further, I believe that the following may help:

int main(){
    point   *points = malloc(sizeof(point));
    if (points == NULL){
        printf("Memory allocation failed.\n");
        return 1;
    }
    other_stuff(&points); /* Send address of points */
    free(points);
    return 0;
}
void other_stuff(point (*ppoints)[]){
    //stuff
    realloc(*ppoints, number*sizeof(point))  /* If a new storage area is assigned by     realloc the original points location will be updated in main */

}