











Estude fácil! Tem muito documento disponível na Docsity
Ganhe pontos ajudando outros esrudantes ou compre um plano Premium
Prepare-se para as provas
Estude fácil! Tem muito documento disponível na Docsity
Prepare-se para as provas com trabalhos de outros alunos como você, aqui na Docsity
Encontra documentos específicos para os exames da tua universidade
Prepare-se com as videoaulas e exercícios resolvidos criados a partir da grade da sua Universidade
Responda perguntas de provas passadas e avalie sua preparação.
Ganhe pontos para baixar
Ganhe pontos ajudando outros esrudantes ou compre um plano Premium
Apostila sobre lista encadeada
Tipologia: Notas de estudo
1 / 19
Esta página não é visível na pré-visualização
Não perca as partes importantes!












#define MAX 1000 int vet[MAX];
10.1. Lista encadeada
Numa lista encadeada, para cada novo elemento inserido na estrutura, alocamos um
espaço de memória para armazená-lo. Desta forma, o espaço total de memória gasto
pela estrutura é proporcional ao número de elementos nela armazenado. No entanto, não
podemos garantir que os elementos armazenados na lista ocuparão um espaço de
memória contíguo, portanto não temos acesso direto aos elementos da lista. Para que
seja possível percorrer todos os elementos da lista, devemos explicitamente guardar o
encadeamento dos elementos, o que é feito armazenando-se, junto com a informação de
cada elemento, um ponteiro para o próximo elemento da lista. A Figura 9.2 ilustra o
arranjo da memória de uma lista encadeada.
A estrutura consiste numa seqüência encadeada de elementos, em geral chamados de
nós da lista. A lista é representada por um ponteiro para o primeiro elemento (ou nó).
Do primeiro elemento, podemos alcançar o segundo seguindo o encadeamento, e assim
por diante. O último elemento da lista aponta para NULL , sinalizando que não existe um
próximo elemento.
Para exemplificar a implementação de listas encadeadas em C, vamos considerar um
exemplo simples em que queremos armazenar valores inteiros numa lista encadeada. O
nó da lista pode ser representado pela estrutura abaixo:
struct lista { int info; struct lista* prox; };
typedef struct lista Lista;
Devemos notar que trata-se de uma estrutura auto-referenciada, pois, além do campo
que armazena a informação (no caso, um número inteiro), há um campo que é um
ponteiro para uma próxima estrutura do mesmo tipo. Embora não seja essencial, é uma
boa estratégia definirmos o tipo Lista como sinônimo de struct lista , conforme
ilustrado acima. O tipo Lista representa um nó da lista e a estrutura de lista encadeada
é representada pelo ponteiro para seu primeiro elemento (tipo Lista*).
Considerando a definição de Lista , podemos definir as principais funções necessárias
para implementarmos uma lista encadeada.
Função de inicialização
A função que inicializa uma lista deve criar uma lista vazia, sem nenhum elemento.
Como a lista é representada pelo ponteiro para o primeiro elemento, uma lista vazia é
10.1.
ULL Info1 Info2 Info
prim
int main (void) { Lista* l; /* declara uma lista não inicializada / l = inicializa(); / inicializa lista como vazia / l = insere(l, 23); / insere na lista o elemento 23 / l = insere(l, 45); / insere na lista o elemento 45 */ ... return 0; }
Observe que não podemos deixar de atualizar a variável que representa a lista a cada
inserção de um novo elemento.
Função que percorre os elementos da lista
Para ilustrar a implementação de uma função que percorre todos os elementos da lista,
vamos considerar a criação de uma função que imprima os valores dos elementos
armazenados numa lista. Uma possível implementação dessa função é mostrada a
seguir.
/* função imprime: imprime valores dos elementos / void imprime (Lista l) { Lista* p; /* variável auxiliar para percorrer a lista */ for (p = l; p != NULL; p = p->prox) printf(“info = %d\n”, p->info); }
Função que verifica se lista está vazia
Pode ser útil implementarmos uma função que verifique se uma lista está vazia ou não.
A função recebe a lista e retorna 1 se estiver vazia ou 0 se não estiver vazia. Como
sabemos, uma lista está vazia se seu valor é NULL. Uma implementação dessa função é
mostrada a seguir:
/* função vazia: retorna 1 se vazia ou 0 se não vazia / int vazia (Lista l) { if (l == NULL) return 1; else return 0; }
Essa função pode ser re-escrita de forma mais compacta, conforme mostrado abaixo:
/* função vazia: retorna 1 se vazia ou 0 se não vazia / int vazia (Lista l) { return (l == NULL); }
Função de busca
Outra função útil consiste em verificar se um determinado elemento está presente na
lista. A função recebe a informação referente ao elemento que queremos buscar e
fornece como valor de retorno o ponteiro do nó da lista que representa o elemento. Caso
o elemento não seja encontrado na lista, o valor retornado é NULL.
/* função busca: busca um elemento na lista / Lista busca (Lista* l, int v) { Lista* p; for (p=l; p!=NULL; p=p->prox) if (p->info == v) return p; return NULL; /* não achou o elemento */ }
Função que retira um elemento da lista
Para completar o conjunto de funções que manipulam uma lista, devemos implementar
uma função que nos permita retirar um elemento. A função tem como parâmetros de
entrada a lista e o valor do elemento que desejamos retirar, e deve retornar o valor
atualizado da lista, pois, se o elemento removido for o primeiro da lista, o valor da lista
deve ser atualizado.
A função para retirar um elemento da lista é mais complexa. Se descobrirmos que o
elemento a ser retirado é o primeiro da lista, devemos fazer com que o novo valor da
lista passe a ser o ponteiro para o segundo elemento, e então podemos liberar o espaço
alocado para o elemento que queremos retirar. Se o elemento a ser removido estiver no
meio da lista, devemos fazer com que o elemento anterior a ele passe a apontar para o
elemento seguinte, e então podemos liberar o elemento que queremos retirar. Devemos
notar que, no segundo caso, precisamos do ponteiro para o elemento anterior para
podermos acertar o encadeamento da lista. As Figuras 9.4 e 9.5 ilustram as operações de
remoção.
Uma possível implementação da função para retirar um elemento da lista é mostrada a
seguir. Inicialmente, busca-se o elemento que se deseja retirar, guardando uma
referência para o elemento anterior.
10.3.
ULL
Info1 Info2 Info
prim
10.4.
ULL
Info1 Info2 Info
prim
Um programa que ilustra a utilização dessas funções é mostrado a seguir.
#include <stdio.h>
int main (void) { Lista* l; /* declara uma lista não iniciada / l = inicializa(); / inicia lista vazia / l = insere(l, 23); / insere na lista o elemento 23 / l = insere(l, 45); / insere na lista o elemento 45 / l = insere(l, 56); / insere na lista o elemento 56 / l = insere(l, 78); / insere na lista o elemento 78 / imprime(l); / imprimirá: 78 56 45 23 / l = retira(l, 78); imprime(l); / imprimirá: 56 45 23 / l = retira(l, 45); imprime(l); / imprimirá: 56 23 */ libera(l); return 0; }
Mais uma vez, observe que não podemos deixar de atualizar a variável que representa a
lista a cada inserção e a cada remoção de um elemento. Esquecer de atribuir o valor de
retorno à variável que representa a lista pode gerar erros graves. Se, por exemplo, a
função retirar o primeiro elemento da lista, a variável que representa a lista, se não fosse
atualizada, estaria apontando para um nó já liberado. Como alternativa, poderíamos
fazer com que as funções insere e retira recebessem o endereço da variável que
representa a lista. Nesse caso, os parâmetros das funções seriam do tipo ponteiro para
lista ( Lista** l ) e seu conteúdo poderia ser acessado/atualizado de dentro da função
usando o operador conteúdo (*l).
Manutenção da lista ordenada
A função de inserção vista acima armazena os elementos na lista na ordem inversa à
ordem de inserção, pois um novo elemento é sempre inserido no início da lista. Se
quisermos manter os elementos na lista numa determinada ordem, temos que encontrar
a posição correta para inserir o novo elemento. Essa função não é eficiente, pois temos
que percorrer a lista, elemento por elemento, para acharmos a posição de inserção. Se a
ordem de armazenamento dos elementos dentro da lista não for relevante, optamos por
fazer inserções no início, pois o custo computacional disso independe do número de
elementos na lista.
No entanto, se desejarmos manter os elementos em ordem, cada novo elemento deve ser
inserido na ordem correta. Para exemplificar, vamos considerar que queremos manter
nossa lista de números inteiros em ordem crescente. A função de inserção, neste caso,
tem a mesma assinatura da função de inserção mostrada, mas percorre os elementos da
lista a fim de encontrar a posição correta para a inserção do novo. Com isto, temos que
saber inserir um elemento no meio da lista. A Figura 9.6 ilustra a inserção de um
elemento no meio da lista.
Conforme ilustrado na figura, devemos localizar o elemento da lista que irá preceder o
elemento novo a ser inserido. De posse do ponteiro para esse elemento, podemos
encadear o novo elemento na lista. O novo apontará para o próximo elemento na lista e
o elemento precedente apontará para o novo. O código abaixo ilustra a implementação
dessa função. Neste caso, utilizamos uma função auxiliar responsável por alocar
memória para o novo nó e atribuir o campo da informação.
/* função auxiliar: cria e inicializa um nó / Lista cria (int v) { Lista* p = (Lista*) malloc(sizeof(Lista)); p->info = v; return p; }
/* função insere_ordenado: insere elemento em ordem / Lista insere_ordenado (Lista* l, int v) { Lista* novo = cria(v); /* cria novo nó / Lista ant = NULL; /* ponteiro para elemento anterior / Lista p = l; /* ponteiro para percorrer a lista*/
/* procura posição de inserção */ while (p != NULL && p->info < v) { ant = p; p = p->prox; }
/* insere elemento / if (ant == NULL) { / insere elemento no início / novo->prox = l; l = novo; } else { / insere elemento no meio da lista */ novo->prox = ant->prox; ant->prox = novo; } return l; }
Devemos notar que essa função, analogamente ao observado para a função de remoção,
também funciona se o elemento tiver que ser inserido no final da lista.
10.5.
ULL
Info1 Info2 Info
prim
Novo
void libera_rec (Lista* l) { if (!vazia(l)) { libera_rec(l->prox); free(l); } }
Exercício: Implemente uma função que verifique se duas listas encadeadas são iguais.
Duas listas são consideradas iguais se têm a mesma seqüência de elementos. O
protótipo da função deve ser dado por: int igual (Lista* l1, Lista* l2);
Exercício: Implemente uma função que crie uma cópia de uma lista encadeada. O
protótipo da função deve ser dado por: Lista* copia (Lista* l);
10.3. Listas genéricas
Um nó de uma lista encadeada contém basicamente duas informações: o encadeamento
e a informação armazenada. Assim, a estrutura de um nó para representar uma lista de
números inteiros é dada por:
struct lista { int info; struct lista *prox; }; typedef struct lista Lista;
Analogamente, se quisermos representar uma lista de números reais, podemos definir a
estrutura do nó como sendo:
struct lista { float info; struct lista *prox; }; typedef struct lista Lista;
A informação armazenada na lista não precisa ser necessariamente um dado simples.
Podemos, por exemplo, considerar a construção de uma lista para armazenar um
conjunto de retângulos. Cada retângulo é definido pela base b e pela altura h. Assim, a
estrutura do nó pode ser dada por:
struct lista { float b; float h; struct lista *prox; }; typedef struct lista Lista;
Esta mesma composição pode ser escrita de forma mais clara se definirmos um tipo
adicional que represente a informação. Podemos definir um tipo Retangulo e usá-lo
para representar a informação armazenada na lista.
struct retangulo { float b; float h; }; typedef struct retangulo Retangulo;
struct lista { Retangulo info; struct lista *prox; }; typedef struct lista Lista;
Aqui, a informação volta a ser representada por um único campo ( info ), que é uma
estrutura. Se p fosse um ponteiro para um nó da lista, o valor da base do retângulo
armazenado nesse nó seria acessado por: p->info.b.
Ainda mais interessante é termos o campo da informação representado por um ponteiro
para a estrutura, em vez da estrutura em si.
struct retangulo { float b; float h; }; typedef struct retangulo Retangulo;
struct lista { Retangulo *info; struct lista *prox; }; typedef struct lista Lista;
Neste caso, para criarmos um nó, temos que fazer duas alocações dinâmicas: uma para
criar a estrutura do retângulo e outra para criar a estrutura do nó. O código abaixo ilustra
uma função para a criação de um nó.
Lista* cria (void) { Retangulo* r = (Retangulo) malloc(sizeof(Retangulo)); Lista p = (Lista*) malloc(sizeof(Lista)); p->info = r; p->prox = NULL; return p; }
Naturalmente, o valor da base associado a um nó p seria agora acessado por: p->info-
b. A vantagem dessa representação (utilizando ponteiros) é que, independente da
informação armazenada na lista, a estrutura do nó é sempre composta por um ponteiro
para a informação e um ponteiro para o próximo nó da lista.
A representação da informação por um ponteiro nos permite construir listas
heterogêneas, isto é, listas em que as informações armazenadas diferem de nó para nó.
Diversas aplicações precisam construir listas heterogêneas, pois necessitam agrupar
elementos afins mas não necessariamente iguais. Como exemplo, vamos considerar uma
aplicação que necessite manipular listas de objetos geométricos planos para cálculos de
áreas. Para simplificar, vamos considerar que os objetos podem ser apenas retângulos,
triângulos ou círculos. Sabemos que as áreas desses objetos são dadas por:
2 2
b h r = b h t = = p
/* Cria um nó com um retângulo, inicializando os campos base e altura / ListaGen cria_ret (float b, float h) { Retangulo* r; ListaGen* p;
/* aloca retângulo / r = (Retangulo) malloc(sizeof(Retangulo)); r->b = b; r->h = h;
/* aloca nó / p = (ListaGen) malloc(sizeof(ListaGen)); p->tipo = RET; p->info = r; p->prox = NULL;
return p; }
/* Cria um nó com um triângulo, inicializando os campos base e altura / ListaGen cria_tri (float b, float h) { Triangulo* t; ListaGen* p;
/* aloca triângulo / t = (Triangulo) malloc(sizeof(Triangulo)); t->b = b; t->h = h;
/* aloca nó / p = (ListaGen) malloc(sizeof(ListaGen)); p->tipo = TRI; p->info = t;
p->prox = NULL; return p; }
/* Cria um nó com um círculo, inicializando o campo raio / ListaGen cria_cir (float r) { Circulo* c; ListaGen* p;
/* aloca círculo / c = (Circulo) malloc(sizeof(Circulo)); c->r = r;
/* aloca nó / p = (ListaGen) malloc(sizeof(ListaGen)); p->tipo = CIR; p->info = c; p->prox = NULL;
return p; }
Uma vez criado o nó, podemos inseri-lo na lista como já vínhamos fazendo com nós de
listas homogêneas. As constantes simbólicas que representam os tipos dos objetos
podem ser agrupadas numa enumeração (ver seção 7.5):
enum { RET, TRI, CIR };
Manipulação de listas heterogêneas
Para exemplificar a manipulação de listas heterogêneas, considerando a existência de
uma lista com os objetos geométricos apresentados acima, vamos implementar uma
função que forneça como valor de retorno a maior área entre os elementos da lista. Uma
implementação dessa função é mostrada abaixo, onde criamos uma função auxiliar que
calcula a área do objeto armazenado num determinado nó da lista:
#define PI 3.
/* função auxiliar: calcula área correspondente ao nó */ float area (ListaGen p) { float a; / área do elemento */
switch (p->tipo) {
case RET: { /* converte para retângulo e calcula área */ Retangulo r = (Retangulo) p->info; a = r->b * r->h; } break;
case TRI: { /* converte para triângulo e calcula área */ Triangulo t = (Triangulo) p->info; a = (t->b * t->h) / 2; } break;
case CIR: { /* converte para círculo e calcula área */ Circulo *c = (Circulo)p->info; a = PI * c->r * c->r; } break; } return a; }
/* Função para cálculo da maior área / float max_area (ListaGen l) { float amax = 0.0; /* maior área / ListaGen p; for (p=l; p!=NULL; p=p->prox) { float a = area(p); /* área do nó */ if (a > amax) amax = a; } return amax; }
elemento. A lista pode ser representada por um ponteiro para um elemento inicial
qualquer da lista. A Figura 9.7 ilustra o arranjo da memória para a representação de uma
lista circular.
Para percorrer os elementos de uma lista circular, visitamos todos os elementos a partir
do ponteiro do elemento inicial até alcançarmos novamente esse mesmo elemento. O
código abaixo exemplifica essa forma de percorrer os elementos. Neste caso, para
simplificar, consideramos uma lista que armazena valores inteiros. Devemos salientar
que o caso em que a lista é vazia ainda deve ser tratado (se a lista é vazia, o ponteiro
para um elemento inicial vale NULL).
void imprime_circular (Lista* l) { Lista* p = l; /* faz p apontar para o nó inicial / / testa se lista não é vazia / if (p) { { / percorre os elementos até alcançar novamente o início / do { printf("%d\n", p->info); / imprime informação do nó / p = p->prox; / avança para o próximo nó */ } while (p != l); }
Exercício: Escreva as funções para inserir e retirar um elemento de uma lista circular.
_10.5. Listas duplamente encadeadas_**
A estrutura de lista encadeada vista nas seções anteriores caracteriza-se por formar um
encadeamento simples entre os elementos: cada elemento armazena um ponteiro para o
próximo elemento da lista. Desta forma, não temos como percorrer eficientemente os
elementos em ordem inversa, isto é, do final para o início da lista. O encadeamento
simples também dificulta a retirada de um elemento da lista. Mesmo se tivermos o
ponteiro do elemento que desejamos retirar, temos que percorrer a lista, elemento por
elemento, para encontrarmos o elemento anterior, pois, dado um determinado elemento,
não temos como acessar diretamente seu elemento anterior.
Para solucionar esses problemas, podemos formar o que chamamos de listas
duplamente encadeadas. Nelas, cada elemento tem um ponteiro para o próximo
elemento e um ponteiro para o elemento anterior. Desta forma, dado um elemento,
podemos acessar ambos os elementos adjacentes: o próximo e o anterior. Se tivermos
um ponteiro para o último elemento da lista, podemos percorrer a lista em ordem
inversa, bastando acessar continuamente o elemento anterior, até alcançar o primeiro
elemento da lista, que não tem elemento anterior (o ponteiro do elemento anterior vale
NULL).
Info1 Info2 Info
ini
A Figura 9.8 esquematiza a estruturação de uma lista duplamente encadeada.
Para exemplificar a implementação de listas duplamente encadeadas, vamos novamente
considerar o exemplo simples no qual queremos armazenar valores inteiros na lista. O
nó da lista pode ser representado pela estrutura abaixo e a lista pode ser representada
através do ponteiro para o primeiro nó.
struct lista2 { int info; struct lista2* ant; struct lista2* prox; };
typedef struct Lista2 Lista2;
Com base nas definições acima, exemplificamos a seguir a implementação de algumas
funções que manipulam listas duplamente encadeadas.
Função de inserção
O código a seguir mostra uma possível implementação da função que insere novos
elementos no início da lista. Após a alocação do novo elemento, a função acertar o
duplo encadeamento.
/* inserção no início / Lista2 insere (Lista2* l, int v) { Lista2* novo = (Lista2) malloc(sizeof(Lista2)); novo->info = v; novo->prox = l; novo->ant = NULL; / verifica se lista não está vazia */ if (l != NULL) l->ant = novo; return novo; }
Nessa função, o novo elemento é encadeado no início da lista. Assim, ele tem como
próximo elemento o antigo primeiro elemento da lista e como anterior o valor NULL. A
seguir, a função testa se a lista não era vazia, pois, neste caso, o elemento anterior do
então primeiro elemento passa a ser o novo elemento. De qualquer forma, o novo
elemento passa a ser o primeiro da lista, e deve ser retornado como valor da lista
atualizada. A Figura 9.9 ilustra a operação de inserção de um novo elemento no início
da lista.
prim
Info1 Info2 Info
Uma implementação da função para retirar um elemento é mostrada a seguir:
/* função retira: retira elemento da lista / Lista2 retira (Lista2* l, int v) { Lista2* p = busca(l,v);
if (p == NULL) return l; /* não achou o elemento: retorna lista inalterada */
/* retira elemento do encadeamento */ if (l == p) l = p->prox; else p->ant->prox = p->prox;
if (p->prox != NULL) p->prox->ant = p->ant;
free(p);
return l; }
Lista circular duplamente encadeada
Uma lista circular também pode ser construída com encadeamento duplo. Neste caso, o
que seria o último elemento da lista passa ter como próximo o primeiro elemento, que,
por sua vez, passa a ter o último como anterior. Com essa construção podemos percorrer
a lista nos dois sentidos, a partir de um ponteiro para um elemento qualquer. Abaixo,
ilustramos o código para imprimir a lista no sentido reverso, isto é, percorrendo o
encadeamento dos elementos anteriores.
void imprime_circular_rev (Lista2* l) { Lista2* p = l; /* faz p apontar para o nó inicial / / testa se lista não é vazia / if (p) { { / percorre os elementos até alcançar novamente o início / do { printf("%d\n", p->info); / imprime informação do nó / p = p->ant; / "avança" para o nó anterior */ } while (p != l); }
Exercício: Escreva as funções para inserir e retirar um elemento de uma lista circular
duplamente encadeada.