C链表 - 何时释放已分配的内存

时间:2017-02-03 16:03:09

标签: c memory-management linked-list malloc free

我有一个简单的链表实现,包括push,pop,unshift和shift函数,可根据需要添加/删除数据。我想确保在通过调用pop和shift来检索数据时,我的实现不会泄漏内存。

如何释放已经通过malloc分配的内存,同时还将数据返回给调用者?

list.h

typedef struct _list_cell_t {
    void *data;
    struct _list_cell_t *next;
} list_cell_t;

typedef struct _list_cell_t *list_cell_ptr;

typedef struct {
    int size;
    list_cell_ptr head;
} list_t;

void list_init(list_t *p_list);
void list_free(list_t *p_list);

void list_push(list_t *p_list, void *data);
void list_unshift(list_t *p_list, void *data);

void *list_pop(list_t *p_list);
void *list_shift(list_t *p_list);

list.c

#include "list.h"

#include <stdlib.h>
#include <string.h>

void list_init(list_t *p_list)
{
    memset(p_list, 0, sizeof(list_t));

    p_list->head = NULL;
    p_list->size = 0;
}

void list_free(list_t *p_list)
{
    list_cell_ptr p_cell, p_next;

    p_cell = p_list->head;
    while (p_cell != NULL) {
        p_next = p_cell->next;
        memset(p_cell, 0, sizeof(list_cell_t));
        free(p_cell);
        p_cell = p_next;
    }

    memset(p_list, 0, sizeof(list_t));
}

void list_push(list_t *p_list, void *data)
{
    list_cell_ptr *p_curr_ptr, p_tmp;

    p_tmp = (list_cell_ptr)malloc(sizeof(list_cell_t));
    memset(p_tmp, 0, sizeof(list_cell_t));
    p_tmp->data = data;

    p_curr_ptr = &(p_list->head);
    while (*p_curr_ptr != NULL) {
        p_curr_ptr = &((*p_curr_ptr)->next);
    }

    p_tmp->next = NULL;
    *p_curr_ptr = p_tmp;
    p_list->size++;
}

void list_unshift(list_t *p_list, void *data)
{
    list_cell_ptr *p_curr_ptr, p_tmp;

    p_tmp = (list_cell_ptr)malloc(sizeof(list_cell_t));
    memset(p_tmp, 0, sizeof(list_cell_t));
    p_tmp->data = data;

    p_curr_ptr = &(p_list->head);

    p_tmp->next = *p_curr_ptr;
    *p_curr_ptr = p_tmp;
    p_list->size++;
}

void *list_pop(list_t *p_list)
{
    list_cell_ptr *p_curr_ptr = &(p_list->head);

    while ((*p_curr_ptr)->next != NULL) {
        p_curr_ptr = &((*p_curr_ptr)->next);
    }

    void *ret = (*p_curr_ptr)->data;

    *p_curr_ptr = NULL;
    p_list->size--;
    return ret;
}

void *list_shift(list_t *p_list)
{
    void *ret = p_list->head->data;

    list_cell_ptr p_next = p_list->head->next;

    p_list->head = p_next;
    p_list->size--;

    return ret;
}

3 个答案:

答案 0 :(得分:1)

在覆盖值并松开您指向的地址之前,您应该使用指针释放内存。你使用那个指针来释放内存空间。

void *list_pop(list_t *p_list)
{
    list_cell_ptr *p_curr_ptr = &(p_list->head);

    while ((*p_curr_ptr)->next != NULL) {
        p_curr_ptr = &((*p_curr_ptr)->next);
    }

    void *ret = (*p_curr_ptr)->data;
    free(*p_curr_ptr); //free memory
    *p_curr_ptr = NULL;

    p_list->size--;
    return ret;
}

void *list_shift(list_t *p_list)
{
    void *ret = p_list->head->data;

    list_cell_ptr p_next = p_list->head->next;
    free(p_list->head); //use this to free memory
    p_list->head = p_next;
    p_list->size--;

    return ret;
} 

我希望我回答你的问题

答案 1 :(得分:1)

  

如何释放已经通过malloc分配的内存,同时还将数据返回给调用者?

总的来说,C内存管理的一般规则是,必须始终明确责任在于释放每一块动态分配的内存,并且无论它在哪里,代码必须注意履行所有这些责任而不失败。在您的情况下,将释放为给定列表分配的struct _list_cell_t对象的责任唯一合理的地方是在代码中再次从列表中删除这些对象(pop,{{1} }和shift函数)。

但是,在释放每个此类对象后,您不能再次访问它,因此必须首先存储要在局部变量中返回的free指针。事实上,你已经这样做了。

有很多方法可以实现细节,但我建议这个范例:

  1. 在本地变量中存储指向不再需要的data的指针。
  2. 更新列表的结构以剪切该对象。
  3. 在本地变量中存储指向所需数据的指针。
  4. 通过步骤(1)中记录的指针释放不需要的struct _list_cell_t
  5. 返回数据

答案 2 :(得分:-1)

你可以使用智能指针。

这可以使用C中的结构来完成。我在C中包含了一个基本的例子。它没有实现你拥有的所有函数(只是push和pop)但是它应该让你知道智能指针

#include <stdlib.h>

// the client get's a pointer to this struct NOT the listitem
struct data
{
    int ref_count;
    int *data;
};

// this is only used by the main, push and pop functions
struct listitem
{
    struct listitem *next;
    struct data *the_data;
};

// the client will have to use decrement() when it is finished with the data
int decrement(struct data *list_data)
{
    if(list_data->ref_count > 0)
        list_data->ref_count--;
    if (list_data->ref_count == 0 && list_data->data)
    {
        free(list_data->data);
        list_data->data = 0;
        free(list_data);
        return 0;
    }
    return(list_data->ref_count);
}

// the client can use increment() if it passes the data to another variable
struct data *increment(struct data *list_data)
{
    if (list_data)
        list_data->ref_count++;
    return(list_data);
}

void free_list(struct listitem **pp)
{
    while (*pp)
    {
        struct listitem *temp = (*pp)->next;
        decrement((*pp)->the_data);
        free(*pp);
        (*pp) = temp;
    }
}

void push_list(struct listitem **pp, int *data)
{
    struct listitem *temp = (struct listitem *)malloc(sizeof(struct listitem));
    temp->next = (*pp);
    (*pp) = temp;
    temp->the_data = (struct data *)malloc(sizeof(struct data));
    temp->the_data->data = data;
    temp->the_data->ref_count = 1;
}

struct data *pop_list(struct listitem **pp)
{
    if (*pp)
    {
        struct listitem *temp = (*pp)->next;
        struct data *d = (*pp)->the_data;
        free(*pp);
        (*pp) = temp;
        return(d); // the data is not being freed from memory here
    }
    return 0;
}

int main()
{
    struct listitem *list = 0;
    int i;

    for (i = 0; i < 10; i++)
    {
        int *pi = (int*)malloc(sizeof(int));
        (*pi) = rand();
        push_list(&list, (int *)pi);
    }
    struct data *d[5];
    for (i = 0; i < 5; i++)
        d[i] = pop_list(&list);
    free_list(&list); // the linked list has been destroyed now

    // this would be the client code
    struct data *d2 = increment(d[0]);

    struct data *d3 = increment(d[1]);

    // clean up, delete all but d2 and d3
    for (i = 0; i < 5; i++)
        decrement(d[i]);

    // do something with d2(d[0]) and d3(d[1])

    // clean up time
    decrement(d2); // now this data has been deleted
    decrement(d3); // now this data has been deleted

    return 0;
}

在代码和客户端之间没有其他方法可以共享内存。