Problem Question
To implement Delete Operations on a Linked List. This includes deleting a node at particular position, deleting a node based on value(search and delete first occurrence), deleting all nodes with a specific value(search and delete all occurrences).
Explanation of Problem
In this program we would be implementing a Linked List. Make sure you have a strong understanding of pointers to understand Linked Lists. A linked list is a basic data structure that is used in dynamic memory allocation applications. It comprises of 'nodes' which are linked together to form a sequence of nodes called Linked List. The linkage is done using memory addresses of adjacent nodes (next node in singly linked list, and both next & previous node in doubly linked list).
In this program we use a struct to implement the node of our linked list. We will implement addition function to get some data in linked list before we can perform desired operations on it. Adding a new node to the list means, creating a new node structure, allocating memory to it and linking it to the list.
Code
#include<iostream>
/**@Title: LinkedList v1.6.cpp*
*@Programming Paradigm: Procedural*
*@Language: C++*
*@Compiler: GNU GCC*
*@IDE: Code::Blocks 13.12*
*@Author: Rogue Coder*
*@URL: http://letsplaycoding.blogspot.com/*
*@Date: 09-12-2014*
*/
struct node
{
  int data;
  node* next;
};
node* deleteNodeFirst(node* rootNode);
node* deleteNodeGlobal(node* rootNode);
node* deleteNode(node* rootNode);
void addAtLast(node** rootNode);
void displayList(node* rootNode);
int main()
{
  int choice;
  node *startList = NULL;
  std::cout << "Welcome to LinkedList v1.5" << std::endl << "Made by Rogue Coder" << std::endl;
  do
  {
    std::cout << std::endl << "1 : Add a Node to the list" <<
         std::endl << "2 : Find and Delete First Occurrence" <<
         std::endl << "3 : Find and Delete All Occurrences" <<
         std::endl << "4 : Delete node at specific position" <<
         std::endl << "5 : Display List" <<
         std::endl << "6 : Exit" <<
         std::endl << "Enter your choice : ";
    std::cin>>choice;
    switch(choice)
    {
    case 1:
        addAtLast(&startList);
        break;
    case 2:
        startList = deleteNodeFirst(startList);
        break;
    case 3:
        startList = deleteNodeGlobal(startList);
        break;
    case 4:
        startList = deleteNode(startList);
        break;
    case 5:
        displayList(startList);
        break;
    case 6:
      std::cout<<std::endl<<"Thank you for using LinkedList v1.6"<<std::endl<<"Made by Rogue Coder"
           <<std::endl<<"Press any key to exit"<<std::endl;
      break;
    default:
      std::cout<<"\a\aWrong Choice\a\a"<<std::endl;
      break;
    }
  }
  while(choice != 6);
  std::cin.get();
  return 0;
}
void addAtLast(node** rootNode)
{
   node* newNode = new node;
   std::cout<<std::endl<<"Enter data : ";
   std::cin>>newNode -> data;
   if(*rootNode == NULL)
   {
      *rootNode = newNode;
   }
   else
   {
       node* currentNode = *rootNode;
       while(currentNode->next != NULL)
       {
           currentNode = currentNode->next;
       }
       currentNode -> next = newNode;
   }
   newNode -> next = NULL;
}
node* deleteNodeFirst(node* rootNode)
{
    int searchKey;
    if (rootNode == NULL)
    {
        std::cout<<std::endl<<"\aList Empty\a";
        return NULL;
    }
    std::cout<<"Enter the Value to Delete: ";
    std::cin>>searchKey;
    if(rootNode -> data == searchKey)
    {
        if(rootNode -> next == NULL)
        {
            return NULL;
        }
        return rootNode -> next;
    }
    node* currentNode = rootNode;
    while (currentNode -> next)
    {
        if(searchKey == currentNode -> next -> data)
        {
            currentNode -> next = currentNode -> next -> next;
            return rootNode;
        }
        currentNode = currentNode -> next;
    }
    std::cout<<"Value not Found";
    return rootNode;
}
node* deleteNodeGlobal(node* rootNode)
{
    int searchKey, value_found = 0;
    if (rootNode == NULL)
    {
        std::cout<<std::endl<<"\aList Empty\a";
        return NULL;
    }
    std::cout<<"Enter the Value to Delete: ";
    std::cin>>searchKey;
    while(rootNode -> data == searchKey)
    {
        value_found++;
        if(rootNode -> next == NULL)
        {
            return NULL;
        }
        rootNode = rootNode -> next;
    }
    node *currentNode = rootNode;
    while (currentNode)
    {
        if(currentNode -> next == NULL && value_found == 0)
        {
            std::cout<<"Value not Found!";
            return rootNode;
        }
        while(currentNode -> next != NULL && searchKey == currentNode -> next -> data)
        {
            value_found++;
            currentNode -> next = currentNode -> next -> next;
        }
        currentNode = currentNode -> next;
    }
    return rootNode;
}
node* deleteNode(node* rootNode)
{
    int deletePosition;
    if (rootNode == NULL)
    {
        std::cout<<std::endl<<"\aList Empty\a";
        return NULL;
    }
    std::cout<<"Enter the Position to Delete: ";
    std::cin>>deletePosition;
    if (deletePosition < 1)
    {
        std::cout<<"Position Out of Bounds";
        return rootNode;
    }
    if (deletePosition == 1)
    {
        if(rootNode -> next == NULL)
        {
            return NULL;
        }
        return rootNode -> next;
    }
    if (deletePosition == 2)
    {
        if(rootNode -> next == NULL)
        {
            std::cout<<"Position Out of Bounds";
            return rootNode;
        }
        rootNode -> next = rootNode -> next -> next;
        return rootNode;
    }
    node *currentNode = rootNode;
    int node_number = 1;
    while (node_number < deletePosition - 1 && currentNode)
    {
        node_number++;
        currentNode = currentNode -> next;
        if (currentNode -> next == NULL)
        {
            std::cout<<"Position Out of Bound";
            return rootNode;
        }
    }
    if (currentNode == NULL)
    {
        std::cout<<"Position Out of Bound";
        return rootNode;
    }
    currentNode -> next = currentNode -> next -> next;
    return rootNode;
}
void displayList(node* rootNode)
{
  node *currentNode = rootNode;
  if(currentNode == NULL)
  {
    std::cout<<std::endl<<"\aList Empty\a"<<std::endl;
  }
  else
  {
    std::cout<<std::endl;
    while(currentNode != NULL)
    {
      std::cout<<currentNode->data<<"->";
      currentNode=currentNode->next;
    }
    std::cout<<"End of List"<<std::endl;
  }
}
Explanation of Code
#include <iostream> -> The compiler calls the Preprocessor to include the IOSTREAM(Standard Input / Output Streams Library) header file into the program, thus letting the use of the Standard Input / Output Streams functions like std::cin and std::cout. As per C++11 specification, including <iostream> automatically includes also <ios>, <streambuf>, <istream>, <ostream> and <iosfwd>.
int main() -> The entry point of the program where the execution starts. This function has to be named main. As per the ANSI specification, the return type has to be int. Since the return type is specified as int in my program, I have to use a return statement at the end of my code. So I use return 0 since zero returned from a function, by convention, implies a correct execution of the program. The return values are used to debug the program.
std::cin (extern istream cin) -> Standard Input Stream, and object of class istream. It is generally used with the extraction operator (>>), though we can use member functions like get (cin.get()), read (cin.read()), etc. for the input. The use of extraction operator is much more popular due to the fact that it aids in getting formatted input.
std::cout (extern ostream cout) -> Standard Output Stream, and object of class ostream. It is generally used with the insertion operator (<<), though we can use member functions like write (cout.write()) for the output. The use of insertions operator is much more popular due to the fact that it aids in giving formatted output.
using namespace std; -> In modern IDEs, we have to explicitly write std::cout instead of cout to use the ostream cout object. Namespace std helps in easing off the pain of writing std:: again and again. Though make sure you are not trapped! The classes defined in std should not be redefined by you. So in case you want to define a class 'distance', you can't do so if you have used std namespace. Though you can define 'Distance' (capital D).
std::endl (ostream& endl (ostream& os)) -> This is a function which is used to insert a newline character and flush the stream. Because this function is a manipulator, it is designed to be used alone with no arguments in conjunction with the insertion (<<) operations on output streams.
struct node
{
int data;
node* next;
}; ->
node* startList; -> The global pointer ‘startList’ which we are going to use to point to the first node/root node/ start node of the linked list, we are going to implement in this program.
int choice; -> This variable 'choice' will be used for the user’s choice in the menu driven program.
void addAtLast(node** rootNode)
{
   node* newNode = new node;
   std::cout<<std::endl<<"Enter data : ";
   std::cin>>newNode -> data;
   if(*rootNode == NULL)
   {
      *rootNode = newNode;
   }
   else
   {
       node* currentNode = *rootNode;
       while(currentNode->next != NULL)
       {
           currentNode = currentNode->next;
       }
       currentNode -> next = newNode;
   }
   newNode -> next = NULL;
}
->
{
    int searchKey;
    if (rootNode == NULL)
    {
        std::cout<<std::endl<<"\aList Empty\a";
        return NULL;
    }
    std::cout<<"Enter the Value to Delete: ";
    std::cin>>searchKey;
    if(rootNode -> data == searchKey)
    {
        if(rootNode -> next == NULL)
        {
            return NULL;
        }
        return rootNode -> next;
    }
    node* currentNode = rootNode;
    while (currentNode -> next)
    {
        if(searchKey == currentNode -> next -> data)
        {
            currentNode -> next = currentNode -> next -> next;
            return rootNode;
        }
        currentNode = currentNode -> next;
    }
    std::cout<<"Value not Found";
    return rootNode;
} ->
The function returns the address of the rootNode of the list, thus in the call to the function we are setting 'startList' variable with the value returned from this function.
We first check if the list is empty. This is done by chcking if the rootNode points to NULL. If so, we prompt the same thing to the user, and return NULL.
Next, we grab user input to get the value that we wish to lookup in the list.
If the value exists at the rootNode, and rootNode is the only node in the list, then we return NULL back(i.e., list is emptied). Otherwise, if the value exists on the rootNode, we set rootNode to point to the next node, and returin this to the function call.
Next, if above checks are untrue, we proceed with list traversal.
We store the rootNode address in a temp variable, 'currentNode', and use it to travers through the list, until we find a node, who's successor has the data we are looking for. If we find such node, we make it to point to the node next to its original successor, removing the successor from the list in effect, and return the rootNode address, else we keep traversing. Once the traversal is complete, and we didnt find the data(if we would have had, out progrma would have returned back, in the if clause itself), we prompt the message to the user 'Value not found' and return the rootNode address back to the function call.
node* deleteNodeGlobal(node* rootNode)
{
    int searchKey, value_found = 0;
    if (rootNode == NULL)
    {
        std::cout<<std::endl<<"\aList Empty\a";
        return NULL;
    }
    std::cout<<"Enter the Value to Delete: ";
    std::cin>>searchKey;
    while(rootNode -> data == searchKey)
    {
        value_found++;
        if(rootNode -> next == NULL)
        {
            return NULL;
        }
        rootNode = rootNode -> next;
    }
    node *currentNode = rootNode;
    while (currentNode)
    {
        if(currentNode -> next == NULL && value_found == 0)
        {
            std::cout<<"Value not Found!";
            return rootNode;
        }
        while(currentNode -> next != NULL && searchKey == currentNode -> next -> data)
        {
            value_found++;
            currentNode -> next = currentNode -> next -> next;
        }
        currentNode = currentNode -> next;
    }
    return rootNode;
}
->
The function returns the address of the rootNode of the list, thus in the call to the function we are setting 'startList' variable with the value returned from this function.
We first check if the list is empty. This is done by chcking if the rootNode points to NULL. If so, we prompt the same thing to the user, and return NULL.
Next, we grab user input to get the value that we wish to lookup in the list.
The first while loop: The code checks if the value exists at the rootNode, it resets the rootNode to it's successor. We have used a while loop to take care of streak of values. For example, if the value existed at a series of positions from 1 to n, this loop takes care of all such values.
Though, if we encounter the node to be last node(rootNode -> next == NULL), we return an empty list back(return NULL, resets the rootNode address).
Next, we proceed with list traversal.
We store the rootNode address in a temp variable, 'currentNode', and use it to travers through the list. We are looking for such a node, who's successor contains the data we wish to delete. If we find such node, we mark its successor as the next node of the original successor. Again, we keep doing this, in case we have a sequence of such nodes (while(currentNode -> next != NULL && searchKey == currentNode -> next -> data)). This process deletes all occurrences of the nodes with the user input value.
When we are able to find the value in the list, we also increment the flag value_found so that we can prompt the user in case of unsuccessful search.
If we reach the second last node, and we still don't find any node who's successor has the data we are looking for, and the value_found flag is 0, we prompt the user with a Value not found message.
node* deleteNodeGlobal(node* rootNode)
{
    int searchKey, value_found = 0;
    if (rootNode == NULL)
    {
        std::cout<<std::endl<<"\aList Empty\a";
        return NULL;
    }
    std::cout<<"Enter the Value to Delete: ";
    std::cin>>searchKey;
    while(rootNode -> data == searchKey)
    {
        value_found++;
        if(rootNode -> next == NULL)
        {
            return NULL;
        }
        rootNode = rootNode -> next;
    }
    node *currentNode = rootNode;
    while (currentNode)
    {
        if(currentNode -> next == NULL && value_found == 0)
        {
            std::cout<<"Value not Found!";
            return rootNode;
        }
        while(currentNode -> next != NULL && searchKey == currentNode -> next -> data)
        {
            value_found++;
            currentNode -> next = currentNode -> next -> next;
        }
        currentNode = currentNode -> next;
    }
    return rootNode;
}
->
If the user input is less than 1, that means it is not a valid location, we prompt user with such information and return.
If the user inputs 1, that means the user wants to delete the rootNode. In such case, we return NULL if rootNode is the only node in the list. Else, rootNode is set as the next node and the function returns next node's address to the call.
If the user had input 2, and we didn't have this position available, we prompt the user with message value out of bounds, else we set change the successor of the rootNode as the next node to original successor.
Otherwise, we traverse through the list, until we reach the node before the one we wish to delete. Once we reach there, we replace the next node address, with that of next to next node. If we don't find such node, we prompt position out of bound and return.
void displayList(node* rootNode)
{
  node *currentNode = rootNode;
  if(currentNode == NULL)
  {
    std::cout<<std::endl<<"\aList Empty\a"<<std::endl;
  }
  else
  {
    std::cout<<std::endl;
    while(currentNode != NULL)
    {
      std::cout<<currentNode->data<<"->";
      currentNode=currentNode->next;
    }
    std::cout<<"End of List"<<std::endl;
  }
}
->
do{..}while() -> The program loop which encapsulates the whole program. Until the user chooses to exit the program, the control loops within this.
exit(0); -> This function is used to exit the program with an error code as it's argument. '0' implies normal exit. Other values are used for debugging purposes.
std::cin.get() -> This statement is used to pause our program, until user presses a key. This function is not necessary in your program, I use it to see my outputs at a paused screen. If you use cmd to run your programs, you might not need this. If you use linux/unix you might not need this. Moreover, removing this line of code from this program, doesn't affect the functionality of the program.
No comments:
Post a Comment
Need help?