Pilha
Uma estrutura de dados feita com o princípio de LIFO (Last In First Out) e que é primordial para o funcionamento de sistemas operacionais, linguagens de programação e interpretação de expressões.
Last updated
Uma estrutura de dados feita com o princípio de LIFO (Last In First Out) e que é primordial para o funcionamento de sistemas operacionais, linguagens de programação e interpretação de expressões.
Last updated
A pilha é uma estrutura de dados que funciona que nem uma pilha da vida real.
Nessa enorme pilha de livros que vemos aqui e considerando que você é uma pessoa cuidadosa, só é possível adicionar ou remover livros no topo. As duas operações principais de uma pilha, no sentido computacional, são também essas:
PUSH
: que adiciona um elemento ao topo da pilha
POP
: que remova um elemento no topo da pilha
Este comportamento é conhecido como LIFO (Last In, First Out), ou seja, o primeiro elemento a entrar é o último a sair, o que vemos também na pilha de livros. Também podemos ver que na pilha de livros, só é possível ver a capa do livro que está no topo, assim também é na estrutura de pilha computacionalmente, onde não é possível ver informações sobre qualquer elemento que não seja o do topo.
Apesar do caráter restritivo dessa estrutura de dados, ela se prova bastante valiosa para determinados usos, como avaliação de expressões matemáticas e gerenciamento de execução em sistemas operacionais e linguagens de programação. Esta estrutura de dados também é usada devido às duas operações serem muito rápidas, como podemos ver na tabela abaixo:
Operação
Complexidade de tempo
PUSH
O(1)
POP
O(1)
Existem duas maneiras de implementar pilhas:
Através de listas encadeadas (ver C)
Através de vetores (ver JavaScript ou Python)
Em ambas as abordagens, a ideia é deitarmos essa pilha, mais ou menos como na figura abaixo:
Para o caso de listas encadeadas, a representação comum envolve uma lista sequencial de todos os elementos até o valor NULL. A representação em pilhas é exatamente a mesma, onde os elementos são adicionados ou removidos pelo começo da lista, a fim de garantir a complexidade das operações.
Para o caso de vetores, é importante que os elementos sejam adicionados ou removidos pelo final do vetor, já que se você remove o primeiro elemento, a própria linguagem se encarrega de "empurrar" todos os elementos da memória para frente e, neste caso, você perde a complexidade constante de O(1) no POP
.
Vamos conferir então como podemos implementar uma pilha nas linguagens suportadas por esse solucionário.
Todos os códigos aqui apresentados (inclusive os providenciados por biblioteca) dão erro ao se tentar acessar ou remover um elemento que não existe. Confira se a pilha está vazia antes de tentar remover ou acessar o elemento do topo.
Aqui eu implementei a nossa pilha como uma struct, onde você precisa inicializa-la toda vez que você precisar usá-la. Eu tomei a liberdade de fazer algumas operações extras de acordo com a pilha implementada na biblioteca STL do C++ (ver subseção abaixo).
A biblioteca STL oferece uma implementação de pilha stack<T>
pronta para ser usada, com as seguintes operações:
push
: insere um elemento no topo da pilha
pop
: remove o elemento do topo da pilha
top
: retorna o elemento que está no topo da pilha
size
: retorna quantos elementos tem na pilha
empty
: retorna se a pilha está vazia
Para mais detalhes sobre como funciona cada operação ou sobre como instanciar uma pilha, veja a documentação.
A saída do programa acima será: 30 20 10
A implementação em JavaScript é bastante simples, podendo ser feita com o auxílio de um array onde só é permitido acrescentar e remover a partir do último índice. Com isso, fazemos meio que uma pilha deitada, onde inserimos e removemos sempre o elemento mais à direita.
Vamos usar exatamente o mesmo raciocínio que usamos para a pilha em JavaScript, onde representando a pilha como um vetor, temos que só podemos adicionar ou remover a partir do último índice.
Python implementa fila duplamente-encadeada com a biblioteca collections
, onde você pode usar a classe deque com o mesmo comportamento de uma pilha se você usar somente as operações append
e pop
.
Você pode implementar a função top
guardando o retorno do pop
e fazendo um push
de volta.
A saída do programa acima será: 30 20 10