A apresentação está carregando. Por favor, espere

A apresentação está carregando. Por favor, espere

Listas Encadeadas Circulares Listas Duplamente Encadeadas

Apresentações semelhantes


Apresentação em tema: "Listas Encadeadas Circulares Listas Duplamente Encadeadas"— Transcrição da apresentação:

1 Listas Encadeadas Circulares Listas Duplamente Encadeadas

2 Listas Encadeadas Circulares

3 Motivação para listas duplamente encadeadas e circulares

4 LISTAS CIRCULARES Um dos inconvenientes do emprego de listas encadeadas consiste no caso de, dado um ponteiro para um nó p de uma lista encadeada, não se ter acesso aos nós que precedem o nó p Este inconveniente pode ser contornado com o emprego de listas circulares Listas circulares são listas encadeadas nas quais o ponteiro do último nó deixa de ser aterrado para apontar o primeiro nó da lista

5 LISTAS CIRCULARES Uma lista circular não tem um primeiro nó nem último nó “naturais” Uma convenção útil é apontar o ponteiro externo (“tail” para a lista circular) para o último nó, sendo o conceito de último vindo de tempo de inclusão O ponteiro head torna-se permanentemente aterrado e o ponteiro tail aponta o último elemento da lista Outra alternativa consiste em usar apenas um ponteiro externo head

6 Listas circulares Ponteiros aterrados representam listas circulares vazias

7 Listas circulares – inclusão
Inclusão de um nó com endereço apontado por tmp na frente e na retaguarda de uma lista circular apontada por tail A inclusão na retaguarda é uma inclusão na frente seguida de tail ← tmp tmp tmp tmp

8 Listas circulares – exclusão
Exclusão do nó da frente de uma lista circular na qual tail aponta o último nó com a recuperação da informação desse nó em y

9 Implementação Java

10 Listas circulares – inclusão na frente
// frente será o sucessor de tail public void insertFront(Object item){ Element tmp = new Element (item, null); if (tail = null) tail = tmp; else { tmp.next = tail.next; } tail.next = tmp;

11 Listas circulares – inclusão no final
// frente será o sucessor de tail // depois tail será tail.next public void insertRear(Object item){ Element tmp = new Element (item, null); if (tail = null) tail = tmp; else { tmp.next = tail.next; } tail.next = tmp; tail = tmp; // transformar primeiro em último

12 Listas circulares – exclusão
public void extractLast(Object item) { if (tail = null) throw new IllegalArgumentException ("item not found"); Element tmp = tail.next; item = tmp.datum; tail.next = tmp.next if (tail = tmp) tail = null; }

13 Implementação C++

14 Listas circulares – Inclusão na frente
template <class T> void LinkedList<T>::InsertFront (T const& item) { ListElement<T>* const tmp = new ListElement<T> (item,0); tail->next = tmp; if(tail == 0) tail = tmp; else tmp->next = tail->next; }

15 Listas circulares – Inclusão no final
template <class T> void LinkedList<T>::InsertFront (T const& item) { // frente será o sucessor de tail // depois tail será tail->next ListElement<T>* const tmp = new ListElement<T> (item,0); ptr->next = tmp; if(tail == 0) tail = tmp; else tmp->next = tail->next; } tail->next = tmp; tail = tmp; // transformar primeiro em último

16 Listas circulares – exclusão
template <class T> void LinkedList<T>::ExtractLast (T const& item) { if (tail == 0) throw invalid_argument "item not found"); ListElement<T>* tmp = 0; tmp = tail->next; item = tmp->datum; tail->next = tmp->next if (tail == tmp) tail = 0; delete tmp; }

17 Listas Duplamente Encadeadas

18 LISTAS DUPLAMENTE ENCADEADAS
Alterando-se os nós de listas encadeadas de forma que cada nó passe a contar, além do registro de informação e do ponteiro para o sucessor, também um ponteiro para antecessor se obtém listas duplamente encadeadas

19 LISTAS DUPLAMENTE ENCADEADAS

20 LISTAS DUPLAMENTE ENCADEADAS
Uma propriedade característica das listas duplamente encadeadas é expressa por: p.next.prev = p = p.prev.next EXEMPLO

21 LISTAS DUPLAMENTE ENCADEADAS
Memória

22 LISTAS DUPLAMENTE ENCADEADAS – Inclusão
inclusão do nó apontado por q à direita do nó apontado por p

23 LISTAS DUPLAMENTE ENCADEADAS – Exclusão
exclusão do nó apontado por q

24 Implementação Java

25 Classe DoubleLinkedList
public class DoubleLinkedList { protected Element head; // cabeça protected Element tail; // cauda public final class Element { // elemento do lista Object datum; // dado do elemento ou item Element next; // ponteiro para sucessor Element prev; // ponteiro para antecessor Element(Object datum, Element next, Element prev) { // construtor this.datum = datum; this.next = next; this.prev = prev; } public Object getDatum() // obter dado { return datum; } public Element getNext() // obter sucessor { return next; } public Element getPrev() // obter antecessor { return prev; } // ...

26 Métodos Construtor e purge
public class DoubleLinkedList { protected Element head; protected Element tail; public DoubleLinkedList () // construir uma lista {} public void purge () // esvaziar uma lista head = null; tail = null; } // ...

27 Métodos getHead, getTail e isEmpty
public class DoubleLinkedList { protected Element head; protected Element tail; public Element getHead () // cabeça da lista { return head; } public Element getTail () // cauda da lista { return tail; } public boolean isEmpty () //teste de lista vazia { return head == null; } // ... }

28 Métods getFirst e getLast
public class DoubleLinkedList { protected Element head; protected Element tail; public Object getFirst() { // obter o primeiro elemento //da lista if(head == null) throw new ContainerEmptyException(); return head.datum; } public Object getLast() { // obter o último elemento //da lista if(tail == null) return tail.datum; // ...

29 Método prepend public class DoubleLinkedList { protected Element head; protected Element tail; // inluir o objeto item no início da lista corrente public void prepend(Object item) Element tmp = new Element(item, head, null); if(head == null) tail = tmp; head.prev = tmp; head = tmp; } // ...

30 Método append public class DoubleLinkedList { protected Element head; protected Element tail; // incluir o objeto item no final da lista corrente public void append (Object item) Element tmp = new Element (item, null, tail); if(head == null) head = tmp; else tail.next = tmp; tail = tmp; } // ...

31 Método assign public class DoubleLinkedList { protected Element head; protected Element tail; // a lista corrente será uma cópia da lista list public void assign(DoubleLinkedList list) { if(list != this) purge (); for(Element ptr = list.head; ptr != null; ptr = ptr.next) append (ptr.datum); } // ...

32 Método extract (1) public class DoubleLinkedList { protected Element head; protected Element tail; // Exclui objeto item public void extract(Object item) Element ptr = head; while(ptr != null && ptr.datum != item) ptr = ptr.next; } if(ptr == null) throw new IllegalArgumentException ("item not found"); if(head == tail) head = null; tail = null;

33 Método extract (2) { ptr.next.prev = null; head = ptr.next; } else
if(ptr == head) // exclui head { ptr.next.prev = null; head = ptr.next; } else if(ptr == tail) // exclui tail ptr.prev.next = null; tail = ptr.prev; ptr.prev.next = ptr.next; ptr.next.prev = ptr.prev; } // fim do extract } // fim da classe DoubleLinkedList

34 Método insertAfter // Inclui item depois do objeto corrente public void insertAfter(Object item) { Element tmp = new Element (item, next, this); if(tail == this){ this.next = tmp; tail = tmp; } else { this.next.prev = tmp;

35 Método insertBefore // Inclui item antes do objeto corrente public void insertBefore(Object item) { Element tmp = new Element(item, this, this.prev); if(head == this) { this.prev = tmp; head = tmp; } else { prev.next = tmp; // ...

36 Implementação C++

37 Definição das Classes DoubleLinkedList<T> e ListElement<T> (1)
template <class T> class DoubleLinkedList; class ListElement { T datum; ListElement* next; ListElement* prev; ListElement (T const&, ListElement*); public: T const& Datum () const; ListElement const* Next () const; ListElement const* Prev() const: friend DoubleLinkedList<T>; };

38 Definição das Classes DoubleLinkedList<T> e ListElement<T> (2)
template <class T> class DoubleLinkedList { ListElement<T>* head; ListElement<T>* tail; public: DoubleLinkedList (); ~DoubleLinkedList (); DoubleLinkedList (DoubleLinkedList const&); DoubleLinkedList& operator = (DoubleLinkedList const&);

39 Definição das Classes DoubleLinkedList<T> e ListElement<T> (3)
ListElement<T> const* Head () const; ListElement<T> const* Tail () const; bool IsEmpty () const; T const& First () const; T const& Last () const; void Prepend (T const&); void Append (T const&); void Extract (T const&); void Purge (); void InsertAfter (ListElement<T> const*, T const&); void InsertBefore (ListElement<T> const*, T const&); };

40 Definição de Funções Membro da Classe ListElement<T>
template <class T> ListElement<T>::ListElement ( T const& _datum, ListElement<T>* _next, ListElement<T>* _prev) : datum (_datum), next (_next), prev (_prev) {} T const& ListElement<T>::Datum () const { return datum; } ListElement<T> const* ListElement<T>::Next () const { return next; } ListElement<T> const* ListElement<T>::Prev () const { return prev; }

41 Construtor Default da Classe ListElement<T>
template <class T> DoubleLinkedList<T>::DoubleLinkedList () : head (0), tail (0) {}

42 Definição de Funções Membro Destrutor e Purge da Classe ListElement<T>
template <class T> void DoubleLinkedList<T>::Purge () { while (head != 0) ListElement<T>* const tmp = head; head = head->next; delete tmp; } tail = 0; DoubleLinkedList<T>::~DoubleLinkedList () { Purge (); }

43 Definição de Funções Accessor da Classe ListElement<T>
template <class T> ListElement<T> const* DoubleLinkedList<T>::Head () const { return head; } ListElement<T> const* DoubleLinkedList<T>::Tail () const { return tail; } bool DoubleLinkedList<T>::IsEmpty () const { return head == 0; }

44 Definição de Funções First e Last da Classe ListElement<T>
template <class T> T const& DoubleLinkedList<T>::First () const { if(head == 0) throw domain_error ("list is empty"); return head->datum; } T const& DoubleLinkedList<T>::Last () const if(tail == 0) return tail->datum;

45 Definição de Função Prepend da Classe ListElement<T>
template <class T> void DoubleLinkedList<T>::Prepend (T const& item) { ListElement<T>* const tmp = new ListElement<T> (item, head, 0); if(head == 0) tail = tmp; head->prev = tmp; head = tmp; }

46 Definição de Função Append da Classe ListElement<T>
template <class T> void DoubleLinkedList<T>::Append (T const& item) { ListElement<T>* const tmp = new ListElement<T> (item, 0, tail); if(head == 0) head = tmp; else tail->next = tmp; tail = tmp; }

47 Definição de Função Construtor Copy da Classe ListElement<T> (1)
template <class T> DoubleLinkedList<T>::DoubleLinkedList (DoubleLinkedList<T> const& DoubleLinkedList) : head (0), tail (0), { ListElement<T> const* ptr; for(ptr = DoubleLinkedList.head; ptr != 0; ptr = ptr->next) Append (ptr->datum); }

48 Definição de Função Construtor Copy da Classe ListElement<T> (2)
template <class T> DoubleLinkedList<T>& DoubleLinkedList<T>::operator = (DoubleLinkedList<T> const& DoubleLinkedList) { if(&DoubleLinkedList != this) Purge (); ListElement<T> const* ptr; for(ptr = DoubleLinkedList.head; ptr != 0; ptr = ptr->next) Append (ptr->datum); } return *this;

49 Definição de Função Extract da Classe ListElement<T> (1)
template <class T> void DoubleLinkedList<T>::Extract (T const& item) { ListElement<T>* ptr = head; while(ptr != 0 && ptr->datum != item) ptr = ptr->next; } if(ptr == 0) throw invalid_argument ("item not found"); if(head == tail) // exclui head head= 0; tail = 0;

50 Definição de Função Extract da Classe ListElement<T> (2)
if(ptr == head) // exclui head { ptr->next->prev = 0; head = ptr->next; } else if(ptr == tail) //exclui tail ptr->prev->next = 0; tail = prevPtr; ptr->prev->next = ptr->next; ptr->next->prev = ptr->prev; delete ptr;

51 Definição da Função InsertAfter da Classe ListElement<T>
template <class T> void DoubleLinkedList<T>::InsertAfter (ListElement<T> const* arg,T const& item) { ListElement<T>* ptr = const_cast<ListElement<T>*> (arg); if(ptr == 0) throw invalid_argument ("invalid position"); ListElement<T>* const tmp = new ListElement<T> (item, ptr->next, ptr); ptr->next = tmp; if(tail == ptr) tail = tmp; else ptr->next->prev = tmp; }

52 Definição da Função InsertBefore da Classe ListElement<T> (1)
template <class T> void LinkedList<T>::InsertBefore (ListElement<T> const* arg, T const& item) { ListElement<T>* ptr = const_cast<ListElement<T>*> (arg); if(ptr == 0) throw invalid_argument ("invalid position"); ListElement<T>* const tmp = new ListElement<T> (item, ptr, ptr->prev);

53 Definição da Função InsertBefore da Classe ListElement<T> (2)
if(head == ptr) {ptr->prev = tmp; head = tmp; } else { ptr->prev = tmp; ptr->prev->next = tmp;


Carregar ppt "Listas Encadeadas Circulares Listas Duplamente Encadeadas"

Apresentações semelhantes


Anúncios Google