#include<bits/stdc++.h>
//#include<leetcode.h>
#define null NULL
#define endl '\n'
#define ll long long
using namespace std;
#define goF(i, a, b) for(int i=a; i<b; i++)
#define goB(i, a, b) for(int i=a; i>=b; i--)
class Node{
public:
int data;
Node* next;
Node(int data){
this->data = data;
this->next = null;
}
~Node(){//Iterative Destructor
delete next;
}
};
Node* takeInput(){
int data; cin>>data;
Node* head = null, *tail = null;
while(data != -1){
Node* newNode = new Node(data);
if(head == null){
head = newNode;
tail = newNode;
}else{
tail->next = newNode;
tail = newNode;
}
cin>>data;
}
return head;
}
Node* createLinkedList(vector<int>& arr){
int pos = 0;
Node* head = null, *tail = null;
while(pos < (int)arr.size()){
Node* newNode = new Node(arr[pos]);
if(head == null){
head = newNode;
tail = newNode;
}else{
tail->next = newNode;
tail = newNode;
}
pos++;
}
return head;
}
void deleteLinkedList(Node* &head){
Node* current = head;
while(current != null){
Node* next = current->next;
delete current;
current = next;
}
delete head;
}
void print_iter(Node* head){
Node* current = head;
while(current != null){
cout<<current->data<<' ';
current = current->next;
}
}
int length_iter(Node* head){
int count = 0;
Node* current = head;
while(current != null){
current = current->next;
count++;
}
return count;
}
Node* evenAfterOdd(Node* head){
if(head == null) return head;
Node* oddHead = null, *oddTail = null, *evenHead = null, *evenTail = null;
Node* current = head;
//Start traversing the Linked List
while(current != null){
//Node* newNode = new Node(current->data);
Node* newNode = new Node(current->data);
if(current->data % 2 == 1){
if(oddHead == null){
oddHead = newNode;
oddTail = newNode;
}else{
oddTail->next = newNode;
oddTail = newNode;
}
}else{//i.e. the current node's data is EVEN
if(evenHead == null){
evenHead = newNode;
evenTail = newNode;
}else{
evenTail->next = newNode;
evenTail = newNode;
}
}
//head = head->next;
current = current->next;
}
//Now you have 2 linked list one with all odd nodes and one with all the even ones
//But precautious: Security Checks
//1. Check 1: odd one was never updated, in that case return the evenHead
if(oddHead == null){
return evenHead; //evenHead could be empty or non empty as well
}else{//otherwise, connect the one with all ODD nodes with the one with all EVEN nodes
oddTail->next = evenHead;
}
if(evenHead != null){
evenTail->next = null;
}
return oddHead;
}
Node* swapNodes(Node* head, int i, int j){
if(i == j) return head; //if both the positions are same, nothing to be swapped, return original head
int length = length_iter(head);
if(i < 0 or i >= length or j < 0 or j >= length) return head;
// We need 4 pointers, that will serve the following purposes
// c1: node at position i, p1: node previous to c1
// c2: node at position j, p2: node previous to c2
Node *p1 = null, *c1 = null, *p2 = null, *c2 = null;
// Run a traversal module with two pointers 'current' and 'previous'
// to set up (p1, c1), (p2, c2)
int pos = 0;
Node *current = head, *previous = null;
while(current != null and (c1 == null or c2 == null)){
//The reason we're using "and (c1 == null or c2 == null)"
//is because we want to quit traversing the linked list
//as soon as we find both c1 and c2
//or in other words as long as either of them are not set
if(pos == i){
p1 = previous;
c1 = current;
}
else if(pos == j){
p2 = previous;
c2 = current;
}
previous = current; //always update previous first
current = current->next; //followed by current
pos++; //and then increase your counter
}
// Once you place the pointers in their appropriate positions, just change links in 3 easy steps
// 1. Connect p1's next to c2
// 2. Connect p2's next to c1
// 3. Connect c2's next to c1's next and c1's next to c2's next
//1. Connect p1's next to c2
//but keep in mind that there's a possibility that p1 could still be null,
//=> p1 never got updated => i == 0
//So, after swapping c2's gonna be the new head
if(p1 != null) p1->next = c2;
else head = c2;
//2. Connect p2's next to c1 (taking care of the edge case)
//Similarly here as well, if p2 is still null then that => j == 0
//so after swapping c1's gonna be the new head
if(p2 != null) p2->next = c1;
else head = c1;
//3. Connect c2's next to c1's next and c1's next to c2's next
Node* c2sNext = c2->next;
c2->next = c1->next;
c1->next = c2sNext;
return head; //and finally we're done!
}
Node* reverse_iter(Node* head){
Node* current = head, *previous = null;
while(current != null){
Node* currentsNext = current->next;
current->next = previous;
previous = current;
current = currentsNext;
}
return previous;
}
Node* reverse_recur(Node* head){
//Base Case:
if(head == null or head->next == null) return head;
//Call recursion on current's next
Node* reversedHead = reverse_recur(head->next);
//Now Post-Processing:
//2nd Node's gonna be the tail of the reversed linked list (key)
Node* tail = head->next;
tail->next = head; //make it point to head
head->next = null; //vvi to severe the ties
return reversedHead;
}
Node* kReverse(Node* head, int k){//similar to reverse_iter()
if(k == 0 or k == 1) return head;
int count = 0;
Node* prev = null, *current = head;
Node* currentsNext = null; //very important to declare outside as that'll help us determine if there's any node left after reversing the first k set of nodes
//Iteratively reverse the first k set of nodes
while(count < k and current != null){
currentsNext = current->next;
current->next = prev;
prev = current;
current = currentsNext;
count++;
}
//then if there's anymore node left (current's next != null)
//then recursively reverse the remaining list, i.e. call kReverse() on current's next
if(currentsNext != null){
head->next = kReverse(currentsNext, k);
}
return prev;
}
Node* delete_recur(Node* head, int pos){
//Base Case:
if(head == null) return head;
//Pre-Processing: If the deletion is at head, then take care of that
if(pos == 1){
Node* newHead = head->next;
head->next = null;
delete head;
return newHead;
}
//otherwise pass on to recursion
Node* updatedHead = delete_recur(head->next, pos-1);
//then connect it to your exisiting head in the post-processing part
head->next = updatedHead;
return head;
}
Node* delete_iter(Node* head, int pos){
//Take explicit care of deletion @head
if(pos == 1){
Node* newHead = head->next;
head->next = null;
delete head;
return newHead;
}
//Otherwise launch the traversal module, run a loop til pos-2 times
Node* current = head;
int count = 0;
while(count < pos-2 and current != null){
current = current->next;
count++;
}
//Now you're @the previous node to be deleted
//do ya thing!
//But safety check first! (i.e. you haven't come out of the traversal loop because current ran out or currently you're at the last node)
if(current == null or current->next == null) return head;
Node* temp = current->next;
current->next = current->next->next;
temp->next = null; //crucial
delete temp;
return head;
}
Node* skipMdeleteN(Node* head, int M, int N){
//If M is 0, we delete all nodes, so return NULL
if(M == 0 || head == NULL) return NULL;
//If N is 0, we keep all nodes, so return the original head
if(N == 0) return head;
Node* currentNode = head; // Pointer to traverse the linked list
Node* temp = NULL; // Will track the last retained node after skipping M nodes
while(currentNode != NULL) {
int take = 0, skip = 0;
// **Step 1: Skip (Retain) M Nodes**
while(currentNode != NULL && take < M){
if(temp == NULL){
temp = currentNode; //Initialize temp to track the first retained node
}else{
temp->next = currentNode; //Ensure proper linkage
temp = currentNode;
}
currentNode = currentNode->next;
take++;
}
// **Step 2: Delete Next N Nodes**
Node* toBeDeleted = currentNode;
while(toBeDeleted != NULL && skip < N){
Node* nextNode = toBeDeleted->next; //Save reference to next node
toBeDeleted->next = NULL; //Important: Disconnect the node before deleting
delete toBeDeleted; //Free memory
toBeDeleted = nextNode; //Move to next node
skip++;
}
// **Step 3: Connect Last Retained Node to Remaining List**
if(temp != NULL) temp->next = toBeDeleted;
//Move `currentNode` to continue processing the remaining list
currentNode = toBeDeleted;
}
return head; //Return the modified linked list
}
int main(){
// cout<<"Enter data: "<<flush;
// Node* head = takeInput();
// print_iter(head); cout<<"| length: "<<length_iter(head)<<endl;
// //For evenAfterOdd()
// vector<int> nums1 = {1, 4, 5, 2};
// Node* head1 = createLinkedList(nums1);
// print_iter(head1); cout<<"| length: "<<length_iter(head1)<<endl;
// head1 = evenAfterOdd(head1);
// print_iter(head1); cout<<"| length: "<<length_iter(head1)<<endl;
//
// cout<<endl;
// vector<int> nums2 = {1, 11, 3, 6, 8, 0, 9};
// Node* head2 = createLinkedList(nums2);
// print_iter(head2); cout<<"| length: "<<length_iter(head2)<<endl;
// head2 = evenAfterOdd(head2);
// print_iter(head2); cout<<"| length: "<<length_iter(head2)<<endl;
//
// cout<<endl;
// vector<int> nums3 = {10, 20, 30, 40};
// Node* head3 = createLinkedList(nums3);
// print_iter(head3); cout<<"| length: "<<length_iter(head3)<<endl;
// head3 = evenAfterOdd(head3);
// print_iter(head3); cout<<"| length: "<<length_iter(head3)<<endl;
// //For swapNodes()
// vector<int> nums = {1, 2, 3, 4, 5, 6, 7};
// Node* head = createLinkedList(nums);
// print_iter(head); cout<<"| length: "<<length_iter(head)<<endl;
//
// //Do modifications here:
// int i = 1, j = 5;
// head = swapNodes(head, i, j);
// print_iter(head); cout<<"| length: "<<length_iter(head)<<endl;
vector<int> nums = {1, 2, 3, 4, 5, 6, 7, 8, 9, 10};
Node* head = createLinkedList(nums);
print_iter(head); cout<<"| length: "<<length_iter(head)<<endl;
// head = reverse_iter(head);
// print_iter(head); cout<<"| length: "<<length_iter(head)<<endl;
//
// head = reverse_recur(head);
// print_iter(head); cout<<"| length: "<<length_iter(head)<<endl;
//
// int k = 4;
// head = kReverse(head, k);
// print_iter(head); cout<<"| length: "<<length_iter(head)<<endl;
int pos = 8;
// head = delete_iter(head, pos);
// //head = delete_recur(head, pos);
// print_iter(head); cout<<"| length: "<<length_iter(head)<<endl;
int M = 2, N = 3;
head = skipMdeleteN(head, M, N);
print_iter(head); cout<<"| length: "<<length_iter(head)<<endl;
return 0;
}
//11 3 9 6 8 0
/*
Sample Input 1: 1 4 5 2 -1
Sample Output 1: 1 5 4 2
Sample Input 2: 1 11 3 6 8 0 9 -1
Sample Output 2:1 11 3 9 6 8 0
Sample Input 3: 10 20 30 40 -1
Sample Output 3: 10 20 30 40
*/