c - 释放 C 堆栈并移除悬挂指针

标签 c pointers data-structures memory-management destructor

我用 C 语言实现了一个堆栈,使用了一个 stackADT 结构和一组函数:

#ifndef _stack_h
#define _stack_h

// Macros
#define MaxStackSize 100
#include <stdlib.h>
#include <stdio.h>
#include <string.h>

// The type of element that may 
// be stored in the stack
typedef int stackElementT;

// The stackADT represents the abstract type used to store
// the elements that have been pushed
typedef struct stackCDT
{
    // A pointer to an array of elements
    stackElementT* elements;
    // Number of elements on the stack
    int count;
    // Number of elements we can push onto
    // the stack before having to resize
    int size;
}* stackADT;

// This function allocates and returns a new stack, which is
// initially empty... AKA - The Constructor
stackADT NewStack(void)
{
    // The stack to return
    stackADT stack;

    // Instanitate a new stack
    stack = (stackCDT*)(malloc(sizeof(stackCDT)));

    // Start with 0 elements of course
    stack->count = 0;
    // Allocate memory for 50 integers
    stack->elements = (stackElementT*)(malloc(50*sizeof(stackElementT)));
    // Establish the size of the stack
    stack->size = 50;

    return stack;
}


/********* GETTER FUNCTIONS *********/
// Returns the number of elements currently pushed 
// onto the stack
int StackDepth(stackADT stack)
{
    return (stack->count);
}

// This function returns the element a the specified index in
// the stack, where the top is defined as index 0
stackElementT GetStackElement(stackADT stack, int index);

// Function to print contents of stack
void PrintStack(stackADT stack)
{
    int i = 0;
    printf("count = %d\nsize = %d\n",stack->count,stack->size);

    for(i = (stack->count - 1); i >= 0; i--)
    {
        if((i%10 == 0) && (i != 0))
            printf("\n");
        printf("%d\t",*(stack->elements + i));
    }
}


// Functions to determine if stack is empty or full
int StackIsEmpty(stackADT stack)
{
    if(stack->count == 0)
        return 1;
    else
        return 0;
}
int StackIsFull(stackADT stack)
{
    if(stack->count == stack->size)
        return 1;
    else
        return 0;
}


// This function pushes the specified element onto the stack
void Push(stackADT stack, stackElementT element)
{
    // A temporary array that we may use later on
    stackElementT* temp = NULL;
    int oldCount = stack->count;
    int i = 0;

    // If the stack if full we need to do a
    // a transfer, resize, and retransfer, then push
    if(StackIsFull(stack))
    {
        // temp will be the same size as the old stack
        temp = (stackElementT*)(malloc((oldCount)*sizeof(stackElementT)));

        // Now we perform the transfer
        for(i = 0; i < oldCount; i++)
        {
            *(temp + i) = *((stack->elements) + i);
        }

        // Free the old memory
        free(stack->elements);
        stack->elements = NULL;

        // Recreate the stack with a 50% increase in size/capacity
        stack->elements = (stackElementT*)(malloc((3*oldCount/2)*sizeof(stackElementT)));
        // Re-establish the size
        stack->size = 3*oldCount/2;

        // Now we perform the transfer back
        for(i = 0; i < oldCount; i++)
        {
            *((stack->elements) + i) = *(temp + i);
        }

        // Free the temp array and 
        // remove dangling pointer
        free(temp);
        temp = NULL;

        // Now we push the element onto the stack
        *((stack->elements) + oldCount) = element;
        // Increase the count
        stack->count = oldCount + 1;
    }
    // If the stack isn't full
    else
    {
        *((stack->elements) + oldCount) = element;
        stack->count = oldCount + 1;
    }

}

// This function pops the top element from the stack and returns
// that value
stackElementT Pop(stackADT stack);


// This function frees the storage associated with the stack
void FreeStack(stackADT stack)
{
    // Start by freeing the elements on the stack
    // and remove dangling pointers
    free(stack->elements);
    stack->elements = NULL;

    // Finally free the stack
    free(stack);
    stack = NULL;
}

#endif

显然我还没有完全完成(需要弹出功能)。我关心的是底层函数 (FreeStack)。我这样测试了下面的代码:

#include <stdio.h>
#include <stdlib.h>
#include "Stack.h"
#define _CRTDBG_MAP_ALLOC
#include <crtdbg.h>

int main(void)
{
    stackADT stack;
    int i = 0;

    stack = NewStack();
    PrintStack(stack);

    for(i = 0; i < 60; i++)
    {
        Push(stack,i);
    }

    PrintStack(stack);
    FreeStack(stack);

    _CrtDumpMemoryLeaks();

    return 0;
}

_CrtDumpMemoryLeaks() 函数用于 Visual Studio,它指示是否存在内存泄漏。显然,我在调用 FreeStack(stackADT stack) 函数时密封了所有泄漏。然而,堆栈指针仍然持有一个内存地址,这是一个问题,因为 FreeStack 函数应该释放堆栈变量指向的内存并将其设置为 NULL。这发生在函数内部,但是当我在调试过程中返回到主函数时,我看到内存地址仍然存在。我在这里错过了什么?如果我能够释放内存,为什么我不能删除悬挂指针?

最佳答案

您按值而不是按地址将堆栈传递给函数,修改函数以接收 (stackADT *) 就可以了。

澄清:正如 Christian 评论的那样,函数调用和堆栈的使用当然也必须更改(因为现在它是指向指针的指针...)

关于c - 释放 C 堆栈并移除悬挂指针,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/5445886/

相关文章:

c - 节点插入、链表

data-structures - m-way树的实际使用

arrays - 提取 2 个数字 n 次并将加法放回 O(n) 而不是 O(n*log(n))

c - 为什么这个矩阵点积程序在某些情况下会在最后一列打印地址?

c - 为什么在给出 -pedantic 时,GCC 仅提示与 %i 一起使用 unsigned int 参数?

centos7中的上下文切换延迟

C++ : Accessing the contents of a pointer

c - 展开循环并使用矢量化进行独立求和

c - 随机和指针不起作用

c# - C#中用于字符串搜索的最有效的集合类是什么