Pilha estática

A pilha é uma estrutura de dados que segue a filosofia LIFO (Last in, first out). Sua concepção prevê um elemento especial, chamado de topo, que representa o local em que tanto as inserções quanto as remoções de elementos serão realizados. Neste ponto, é imprescindível enfatizar que, em pilhas, remoções e inserções não podem ocorrer em qualquer outra posição que não seja o topo.

A pilha pode ser implementada de forma estática (usando vetores) ou de forma dinâmica, com alocação de memória ajustável conforme a necessidade.

🔋 Declarando a pilha

O código a seguir apresenta a declaração das estruturas necessárias para a criação (e manipulação) de uma pilha implementada de forma estática. Para começar, temos uma estrutura chamada REGISTRO, que pode armazenar diversas variáveis de interesse. Para fins de exemplo, o código apresenta apenas uma variável, chave.

A pilha em si é definida pela estrutura PILHA, que contém registros, um vetor de itens do tipo REGISTRO. O topo, neste caso, é representado por uma variável do tipo inteiro, que representa a posição da pilha utilizada como topo.

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

#define TAM 3

typedef struct{
    int chave;
} REGISTRO;

typedef struct{
    REGISTRO registros[TAM];
    int topo;
} PILHA;

*️⃣ Funções de manipulação da pilha

Como destacado anteriormente, uma estrutura de dados do tipo pilha envolve algumas operações essenciais.

Inicializar a pilha

O procedimento de inicialização da pilha consiste, basicamente, em atribuir o valor 0 à variável topo da pilha. Esse processo de atribuição é necessário porque, ao instanciarmos uma pilha, o valor dessa variável será desconhecido (lembre-se que em C as variáveis não inicializadas possuem lixo de memória como valor).

O processo de inicialização garante que a pilha funcione de forma adequada, já que sua variável topo terá o valor necessário para o funcionamento do algoritmo.

void inicializarPilha(PILHA *p){
    p->topo = 0;
}

Inserir (ou empilhar)

A inserção de elementos numa pilha estática assume que a variável topo representa a posição do vetor em que a inserção deve ser efetuada. Se a pilha estiver vazia, a inserção será na posição 0. Se a pilha tiver um elemento, a inserção será na posição 1 do vetor.

Desse modo, deve-se observar se o valor da variável p->topo é menor que a quantidade de itens da pilha. Isso acontece porque, se a pilha puder armazenar 3 elementos (#define TAM 3), as posições válidas do vetor vão de 0 a 2.

void inserir(PILHA *p, REGISTRO r){
    if(p->topo < TAM){
        p->registros[p->topo] = r;
        p->topo++;
    }
    else{
        printf("Pilha cheia!\n");
    }
}

Remover (ou desempilhar)

A operação de remoção de um elemento da pilha estática é bastante simples: precisamos apenas verificar se a pilha possui ao menos um elemento. Em seguida, decrementamos o valor da variável topo.

void remover(PILHA *p){
    if(p->topo > 0){
        p->topo--;
    }
}

Imprimir

A impressão de uma pilha estática envolve o uso de um laço de repetição, como for ou while. A diferença é que, ao contrário do habitual, o laço deve iniciar no topo da pilha e iterar até o primeiro elemento da pilha, representado pelo índice 0 do vetor.

void imprimir(PILHA p){
    for(int i = p.topo - 1; i >= 0; i--){
        printf("%d\n", p.registros[i].chave);
    }
}

🧑🏻‍💻 Testando a pilha

Para testar a pilha, podemos implementar o main da seguinte forma:

int main(void){
    PILHA p;
    inicializarPilha(&p);
    inserir(&p, (REGISTRO){10});
    inserir(&p, (REGISTRO){20});
    inserir(&p, (REGISTRO){30});
    inserir(&p, (REGISTRO){40});
    remover(&p);
    inserir(&p, (REGISTRO){40});
    imprimir(p);
}