Design your implementation of the linked list. You can choose to use a singly or doubly linked list.
A node in a singly linked list should have two attributes: val and next. val is the value of the current node, and next is a pointer/reference to the next node.
If you want to use the doubly linked list, you will need one more attribute prev to indicate the previous node in the linked list. Assume all nodes in the linked list are 0-indexed.
Implement the MyLinkedList class:
MyLinkedList() Initializes the MyLinkedList object.
int get(int index) Get the value of the index-th node in the linked list. If the index is invalid, return -1.
void addAtHead(int val) Add a node of value val before the first element of the linked list. After the insertion, the new node will be the first node of the linked list.
void addAtTail(int val) Append a node of value val as the last element of the linked list.
void addAtIndex(int index, int val) Add a node of value val before the index-th node in the linked list. If index equals the length of the linked list, the node will be appended to the end of the linked list. If index is greater than the length, the node will not be inserted.
void deleteAtIndex(int index) Delete the index-th node in the linked list, if the index is valid.
MyLinkedListmyLinkedList=newMyLinkedList();myLinkedList.addAtHead(1);myLinkedList.addAtTail(3);myLinkedList.addAtIndex(1,2);// linked list becomes 1->2->3myLinkedList.get(1);// return 2myLinkedList.deleteAtIndex(1);// now the linked list is 1->3myLinkedList.get(1);// return 3
Constraints:
0 <= index, val <= 1000
Please do not use the built-in LinkedList library.
At most 2000 calls will be made to get, addAtHead, addAtTail, addAtIndex and deleteAtIndex.
classMyLinkedList{public:structLinkedNode{intval;LinkedNode*next;LinkedNode(intval):val(val),next(nullptr){}};MyLinkedList(){_dummyHead=newLinkedNode(0);_size=0;}intget(intindex){if(index>=_size||index<0)return-1;LinkedNode*curr=_dummyHead->next;while(index--&&curr)curr=curr->next;if(curr)returncurr->val;return-1;}voidaddAtHead(intval){LinkedNode*newNode=newLinkedNode(val);newNode->next=_dummyHead->next;_dummyHead->next=newNode;_size++;}voidaddAtTail(intval){LinkedNode*newNode=newLinkedNode(val);LinkedNode*curr=_dummyHead;while(curr->next!=nullptr)curr=curr->next;curr->next=newNode;_size++;}voidaddAtIndex(intindex,intval){if(index>_size)return;if(index<0)index=0;LinkedNode*newNode=newLinkedNode(val);LinkedNode*curr=_dummyHead;while(index--&&curr->next)curr=curr->next;newNode->next=curr->next;curr->next=newNode;_size++;}voiddeleteAtIndex(intindex){if(index>=_size||index<0)return;LinkedNode*curr=_dummyHead;while(index--&&curr->next)curr=curr->next;LinkedNode*temp=curr->next;curr->next=curr->next->next;deletetemp;temp=nullptr;_size--;}private:int_size;LinkedNode*_dummyHead;};/** * Your MyLinkedList object will be instantiated and called as such: * MyLinkedList* obj = new MyLinkedList(); * int param_1 = obj->get(index); * obj->addAtHead(val); * obj->addAtTail(val); * obj->addAtIndex(index,val); * obj->deleteAtIndex(index); */
Time complexity: for operations involving index, for the rest;
Space complexity: .
Note: When we insert with index < 0, test case will just treat as insert at index = 0.
classMyLinkedList{public:// Node structure representing each element in the liststructLinkedNode{intval;// Value of the nodeLinkedNode*next;// Pointer to the next node in the listLinkedNode(intval):val(val),next(nullptr){}};// ConstructorMyLinkedList(){_dummyHead=newLinkedNode(0);// Create a dummy head to simplify operations_size=0;// Initialize the size of the list as 0}// Get the value of the index-th node in the linked listintget(intindex){if(index>=_size||index<0)return-1;// Return -1 for invalid indexLinkedNode*curr=_dummyHead->next;// Start from the first real nodewhile(index--)curr=curr->next;// Move to the correct nodereturncurr->val;// Return the value of the node}// Add a node of value val before the first element of the linked listvoidaddAtHead(intval){LinkedNode*newNode=newLinkedNode(val);// Create a new nodenewNode->next=_dummyHead->next;// Point the new node to the first real node_dummyHead->next=newNode;// Insert the new node after dummy head_size++;// Increase the size of the list}// Append a node of value val to the last element of the linked listvoidaddAtTail(intval){LinkedNode*newNode=newLinkedNode(val);// Create a new nodeLinkedNode*curr=_dummyHead;// Start from the dummy headwhile(curr->next!=nullptr)// Traverse to the end of the listcurr=curr->next;curr->next=newNode;// Append the new node_size++;// Increase the size of the list}// Add a node of value val before the index-th node in the linked listvoidaddAtIndex(intindex,intval){if(index>_size)return;// Do nothing if index is greater than the sizeif(index<0)index=0;// Treat negative index as 0LinkedNode*newNode=newLinkedNode(val);// Create a new nodeLinkedNode*curr=_dummyHead;// Start from the dummy headwhile(index--)curr=curr->next;// Move to the correct positionnewNode->next=curr->next;// Insert the new nodecurr->next=newNode;_size++;// Increase the size of the list}// Delete the index-th node in the linked list, if the index is validvoiddeleteAtIndex(intindex){if(index>=_size||index<0)return;// Do nothing if the index is invalidLinkedNode*curr=_dummyHead;// Start from the dummy headwhile(index--)curr=curr->next;// Move to the node before the one to be deletedLinkedNode*temp=curr->next;// Temporarily store the node to be deletedcurr->next=curr->next->next;// Remove the node from the listdeletetemp;// Delete the nodetemp=nullptr;_size--;// Decrease the size of the list}private:int_size;// Size of the linked listLinkedNode*_dummyHead;// Dummy head node to simplify operations};/** * Your MyLinkedList object will be instantiated and called as such: * MyLinkedList* obj = new MyLinkedList(); * int param_1 = obj->get(index); * obj->addAtHead(val); * obj->addAtTail(val); * obj->addAtIndex(index,val); * obj->deleteAtIndex(index); */
classMyLinkedList{public:// Node structure for the linked liststructNode{intval;// Value of the nodeNode*next;// Pointer to the next nodeNode(int_val):val(_val),next(nullptr){}// Constructor to initialize node}*head;// Head pointer of the list// Constructor initializes the linked listMyLinkedList(){head=nullptr;// Initially, the list is empty, so head is null}// Function to get the value of the node at a given indexintget(intindex){if(index<0)return-1;// Return -1 for invalid indexautop=head;// Start from the head of the listfor(inti=0;i<index&&p;i++)p=p->next;// Traverse the list to reach the desired indexif(!p)return-1;// If p is null, index is out of boundsreturnp->val;// Return the value of the node at the index}// Function to add a node at the beginning of the listvoidaddAtHead(intval){autocurr=newNode(val);// Create a new nodecurr->next=head;// Link new node to the current headhead=curr;// Update head to the new node}// Function to add a node at the end of the listvoidaddAtTail(intval){if(!head){// If the list is emptyhead=newNode(val);// Create a new node and set it as head}else{autop=head;// Start from the headwhile(p->next)p=p->next;// Traverse to the last nodep->next=newNode(val);// Add new node at the end}}// Function to add a node at a specific indexvoidaddAtIndex(intindex,intval){if(index<=0){// If index is 0 or negative, add at headaddAtHead(val);}else{intlen=0;for(autop=head;p;p=p->next)len++;// Calculate the length of the listif(index==len){// If index equals the length of the list, add at tailaddAtTail(val);}elseif(index<len){// If index is within boundsautop=head;for(inti=0;i<index-1;i++)p=p->next;// Find the node before the desired indexautocurr=newNode(val);// Create a new nodecurr->next=p->next;// Insert the new node after pp->next=curr;}}}// Function to delete a node at a specific indexvoiddeleteAtIndex(intindex){intlen=0;for(autop=head;p;p=p->next)len++;// Calculate the length of the listif(index>=0&&index<len){// If index is within boundsif(!index){// If index is 0, delete the headhead=head->next;}else{autop=head;for(inti=0;i<index-1;i++)p=p->next;// Find the node before the one to deletep->next=p->next->next;// Delete the node at the index}}}};/** * Your MyLinkedList object will be instantiated and called as such: * MyLinkedList* obj = new MyLinkedList(); * int param_1 = obj->get(index); * obj->addAtHead(val); * obj->addAtTail(val); * obj->addAtIndex(index,val); * obj->deleteAtIndex(index); */
Find the predecessor and the successor of the node to insert. If the node is to be inserted at head, its predecessor is a sentinel head. If the node is to be inserted at the tail, its successor is a sentinel tail. (Use bidirectional search to perform faster.)
Insert the node by changing the links to the next and previous nodes.
classMyLinkedList{public:structDoubleLinkedNode{intval;DoubleLinkedNode*prev;DoubleLinkedNode*next;DoubleLinkedNode(int_val=0):val(_val),prev(nullptr),next(nullptr){}};MyLinkedList(){_size=0;head=newDoubleLinkedNode(0);tail=newDoubleLinkedNode(0);// sentinel (dummy) nodes as pseudo-head and pseudo-tailhead->next=tail;tail->prev=head;}// Destructor to free memory of all nodes including sentinel nodes~MyLinkedList(){autocurr=head;while(curr){autotemp=curr;curr=curr->next;deletetemp;}}intget(intindex){if(index<0||index>=_size)return-1;autocurr=head;if(index+1>_size-index){for(inti=0;i<index+1;i++)curr=curr->next;}else{curr=tail;for(inti=0;i<_size-index;i++)curr=curr->prev;}returncurr->val;}voidaddAtHead(intval){addAtIndex(0,val);// DoubleLinkedNode* pred = head, * succ = head->next;// _size++;// auto toAdd = new DoubleLinkedNode(val);// toAdd->prev = pred;// toAdd->next = succ;// pred->next = toAdd;// succ->prev = toAdd;}voidaddAtTail(intval){addAtIndex(_size,val);// DoubleLinkedNode* succ = tail, * pred = tail->prev;// _size++;// auto toAdd = new DoubleLinkedNode(val);// toAdd->prev = pred;// toAdd->next = succ;// pred->next = toAdd;// succ->prev = toAdd;}voidaddAtIndex(intindex,intval){if(index>_size)return;if(index<0)index=0;DoubleLinkedNode*pred,*succ;if(index<_size-index){pred=head;for(inti=0;i<index;i++)pred=pred->next;succ=pred->next;}else{succ=tail;for(inti=0;i<_size-index;i++)succ=succ->prev;pred=succ->prev;}_size++;autotoAdd=newDoubleLinkedNode(val);toAdd->prev=pred;toAdd->next=succ;pred->next=toAdd;succ->prev=toAdd;}voiddeleteAtIndex(intindex){if(index<0||index>=_size)return;DoubleLinkedNode*pred,*succ;if(index<_size-index){pred=head;for(inti=0;i<index;i++)pred=pred->next;succ=pred->next->next;}else{succ=tail;for(inti=0;i<_size-index-1;i++)succ=succ->prev;pred=succ->prev->prev;}_size--;// Remove the node from the list// pred->next = succ;// succ->prev = pred;autotoDelete=pred->next;pred->next=succ;succ->prev=pred;deletetoDelete;// Free the memory of the node to be deleted}private:int_size;DoubleLinkedNode*head;DoubleLinkedNode*tail;};/** * Your MyLinkedList object will be instantiated and called as such: * MyLinkedList* obj = new MyLinkedList(); * int param_1 = obj->get(index); * obj->addAtHead(val); * obj->addAtTail(val); * obj->addAtIndex(index,val); * obj->deleteAtIndex(index); */