280 likes | 290 Views
Learn about linked lists, a data structure that optimizes memory usage by dynamically allocating and deallocating memory based on the number of elements in use. Compared to arrays, linked lists allow for direct access and have no maximum number of elements.
E N D
Motivation A list data structure that uses only the amount of memory needed for the number of elements in use at a given point in time.
Compared to Arrays Arrays: - easily allocated/deallocated - direct access (accessing element n is done simply by indexing: [n]) Linked Lists: - no maximum number of elements - no waste of memory for unused elements.
Node A Node is a dynamically allocated listelement, implemented as a structure that has at least two members: - key: a (usually) unique identifier for an element. - next: a pointer to a node.
Definition A Linked Listis a linear, dynamically-allocated data structure. It consists of a head which is a pointer to a node. - The head points to the first element of the list; or is NULL when the list is empty. - The next pointer of each node points to the next node in the list; except the last node, where next is NULL.
Class Definitions class node { friend class LList; private: int key; node * next; // circular reference OK }; class LList { private: node * head; };
Algorithms When designing a function that operates on a Linked List, consider if the solution works for: - an empty list - a list of one node - a list of two or more nodes
Algorithms "The Loop that solves all problems!" (well, most anyway) // point p to the first/head node node * p = head; while (p != NULL) { // process this node ... p = p->next; // go to next node } // or NULL
Constructors #include <iostream> node::node() { key = 0; next = NULL; } LList::LList() { head = NULL; }
Print void LList::print() { node * p = head; while (p != NULL) { cout << p->key << " "; p = p-> next; } }
Search // return ptr to found node or // NULL = not found node * LList::search(intsrchKey) { node * p = head; while (p != NULL) { if (p->key == srchKey) return p; p = p->next; } return NULL; //looked at all, not found }
Insert A new node should first be created and populated: void LList::insert(intnewkey) { node * n = new node; //allocate new node n->key = newkey; n->next = NULL; insert(n); // some insertion method } // given pointer to new node
Insert (After) Given: - after: a pointer to a node in the List, after which the new node will be inserted. NULL to insert new head. - n: a pointer to the new node, already allocated and populated.
Insert General Case: Blue arrows/null show values before insertion. Red arrows show links afterinsertion (changes)
Insert New Head Case: red arrows show links after changes. Empty List Case: the same code will work when head is NULL
Insert void LList::insertAfter(node* after, node* n) { if (head == NULL || after == NULL) { n->next = head; // insert as new head = n; // head of list } else { // insert middle/tail of list n->next = after->next; after->next = n; } // note: ORDER of statements is IMPORTANT! }
Insert Checks: - Does it work for Empty List? [head is NULL] - Does it work for a List of 1 element? - as the new head? [after is NULL] - after the current head? [after == head] - Does it work for a list of 2 or more elements? - as the new head? [after is NULL] - as the new tail? (last in list) [after points to tail] - in the middle of the list?
Remove Given: r- a pointer to a node in the List to be removed Removes the node from the list, but does not deallocate it. Alternative: Removes and deallocates the node.
Remove Remove Head case:
Remove General Case: Use b4 to find a pointer to the node before the one to remove
Remove void LList::remove(node * r) { if (r == head) head = head->next; else { // must find node BEFORE one being removed node * b4 = head; while (b4->next != r) { b4 = b4->next; } b4->next = r->next; } r->next = NULL; // detach following node }
Remove Checks: - Assumption: never invoked on Empty List - Assumption: r points to a node actually in the List. - Does it work for a List of 1 element? - Does it work for a list of 2 or more elements? - removing the head? - removing the tail? - removing a node in the middle of the list?
Remove - Defensive Version void LList::remove(node * r) { if (head == NULL) return; // list empty, can't remove if (r == NULL) return; // nothing to remove!? if (r == x.head) x.head= x.head->next; else { // must find node BEFORE one being removed node * b4 = x.head; while (b4->next != r && b4 != NULL) { b4 = b4->next; } if (b4 == NULL) return; // node not in list!! b4->next = r->next; } r->next = NULL; }
Remove Note: r still points to the removed node after remove() is finished. The function invoking remove() may choose to deallocate it or not.
Remove - invocation void LList::fun() { ... node * p; p = ptr to node to remove; remove(p); cout << p->key << " was removed"; delete p; // deallocate removed node ... }
Length // return: count of the nodes in the list void LList::length() { int num = 0; node * p = head; while (p != NULL) { num++; p = p->next; } return num; }