在不同的功能中分配空闲内存?

时间:2012-06-17 12:11:31

标签: c pointers memory-management malloc free

我正在尝试学习C而我正在尝试编写一个基本的堆栈数据结构,但我似乎无法正确获得基本的malloc / free

这是我一直在使用的代码(我只是在这里发布一小部分来说明一个特定问题,而不是总代码,但错误消息是通过在valgrind中运行此示例代码生成的)

#include <stdio.h>
#include <stdlib.h>

typedef struct Entry {
    struct Entry *previous;
    int value;
} Entry;

void destroyEntry(Entry entry);

int main(int argc, char *argv[])
{
    Entry* apple;
    apple = malloc(sizeof(Entry));
    destroyEntry(*(apple));
    return 0;
}

void destroyEntry(Entry entry)
{
    Entry *entry_ptr = &entry;
    free(entry_ptr);
    return;
}

当我使用valgrind运行--leak-check=full --track-origins=yes时,我收到以下错误:

==20674== Invalid free() / delete / delete[] / realloc()
==20674==    at 0x4028E58: free (vg_replace_malloc.c:427)
==20674==    by 0x80485B2: destroyEntry (testing.c:53)
==20674==    by 0x8048477: main (testing.c:26)
==20674==  Address 0xbecc0070 is on thread 1's stack

我认为此错误意味着不允许destroyEntry函数修改main中明确分配的内存。是对的吗?为什么我不能free在另一个函数中main分配的内存{{1}}? (这种行为是否以某种方式特定于主?)

3 个答案:

答案 0 :(得分:51)

每当您将参数传递给函数时,都会生成一个副本,并且该函数适用于该副本。因此,在您的情况下,您正在尝试free原始对象的副本,这没有任何意义。

您应该修改函数以获取指针,然后您可以直接在该指针上调用free

答案 1 :(得分:37)

这是传递值,这意味着创建了副本,因此您尝试释放本地变量entry所在的内存。请注意entry是一个具有自动存储持续时间的对象,当程序超出destroyEntry函数范围时,它所驻留的内存将自动释放。

void destroyEntry(Entry entry)
{
    Entry *entry_ptr = &entry;
    free(entry_ptr);
    return;
}

你的函数应该带一个指针(通过引用传递):

void destroyEntry(Entry *entry)
{
    free(entry);
}

然后,您只需拨打destroyEntry(*(apple));而不是destroyEntry(apple);。请注意,如果没有其他功能与destroyEntry功能相关联,则它是多余的,最好直接调用free(apple)

答案 2 :(得分:9)

这里的其他答案指出了主要问题 - 因为你在main()中调用destroyEntry时取消引用你的苹果,它会通过引用传递,创建一个副本。

即使您知道自己的问题,也可以回到错误并尝试将您所看到的文本与问题联系起来,以便下次出现时您可能更有可能快点出来我发现C和C ++错误有时会显得格外含糊。

通常,当我在释放指针或删除对象时遇到问题时,我喜欢打印出地址,尤其是在我分配地址时,以及当我尝试释放它时。 valgrind已经给你了坏指针的地址,但它有助于将它与一个好指针进行比较。

int main()
{
  Entry * apple;
  apple = malloc(sizeof(Entry));
  printf("apple's address = %p", apple);  // Prints the address of 'apple'
  free(apple);   // You know this will work
}

执行此操作后,您会注意到printf()语句为您提供了一个类似于0x8024712的地址(只是在正常的范围内组成一个地址),但您的valgrind输出给出了0x4028E58。您会注意到它们位于两个非常不同的位置(实际上,“0x4 ......”在堆栈中,而不是malloc()分配的堆,但我假设您刚刚开始使用它不是你的红旗),所以你知道你试图从错误的地方释放记忆,因此“无效的免费()”。

所以从那里你可以对自己说“好吧,不知怎的,我的指针已经腐烂了。”你已经把你的问题归结为一个小的,可编辑的例子,所以从那里解决它不会花费你很长时间。

TL; DR - 遇到与指针相关的错误时,请尝试打印地址或在您喜欢的调试器中找到它们。它通常至少指向正确的方向。

当然,这些都不是为了阻止在Stack Exchange上发布您的问题。数以百计的程序员可能会从你这样做中受益。