C ++
 Computer >> コンピューター >  >> プログラミング >> C ++

C ++を使用して、指定されたサイズのグループでリンクリストを反転します


この記事では、単一リンクリストを扱います。タスクは、kのグループでリストを逆にすることです。例-

Input: 1->2->3->4->5->6->7->8->NULL, K = 3
Output: 3->2->1->6->5->4->8->7->NULL

Input: 1->2->3->4->5->6->7->8->NULL, K = 5
Output: 5->4->3->2->1->8

この問題の場合、頭に浮かぶアプローチの1つは、サブリストのサイズがkに達して続行したときに、リストを追跡し、リストを逆にすることです。

解決策を見つけるためのアプローチ

このアプローチでは、通常、リストをトラバースし、サブリスト内の要素の数をカウントするためのカウンターを保持します。カウンターがkのカウントに達すると、その部分を逆にします。

#include <bits/stdc++.h>
using namespace std;
class Node {
   public:
   int data;
   Node* next;
};
Node* reverse(Node* head, int k) {
   if (!head)
      return NULL;
   Node* curr = head;
   Node* next = NULL;
   Node* prev = NULL;
   int count = 0;
   while (curr != NULL && count < k) { // we reverse the list till our count is less than k
      next = curr->next;
      curr->next = prev;
      prev = curr;
      curr = next;
      count++;
   }
   if (next != NULL) // if our link list has not ended we call reverse function again
      head->next = reverse(next, k);
   return prev;
}
void push(Node** head_ref, int new_data) { // function for pushing data in the list
   Node* new_node = new Node();
   new_node->data = new_data;
   new_node->next = (*head_ref);
   (*head_ref) = new_node;
}
void printList(Node* node) { // function to print linked list

   while (node != NULL) {
      cout << node->data << " ";
      node = node->next;
   }
   cout << "\n";
}
int main() {
   Node* head = NULL;

   int k = 3; // the given k

   push(&head, 8);
   push(&head, 7);
   push(&head, 6);
   push(&head, 5);
   push(&head, 4);
   push(&head, 3);
   push(&head, 2);
   push(&head, 1);

   cout << "Original list \n";
   printList(head);

   head = reverse(head, k); // this function will return us our new head
   cout << "New list \n";
   printList(head);
   return (0);
}

出力

Original list
1 2 3 4 5 6 7 8
New list
3 2 1 6 5 4 8 7

上記のアプローチには、 O(N)の時間計算量があります。 、ここで、Nは指定されたリストのサイズであり、このアプローチは再帰で機能します。このアプローチは、より高い制約に対しても機能します。

上記のコードの説明

このアプローチでは配列をトラバースし、カウンター変数がk未満になるまで配列を反転し続けます。カウンターがkの値に達すると、別の反転関数を呼び出して、このサブリストの最後のノードを次の反転サブリストの最初のノードに接続します。これは再帰によって行われています。

結論

この記事では、再帰を使用して、指定されたサイズのグループでリンクリストを逆にする問題を解決します。また、この問題のC ++プログラムと、この問題を解決するための完全なアプローチ(Normal)についても学びました。同じプログラムを、C、java、python、その他の言語などの他の言語で作成できます。この記事がお役に立てば幸いです。


  1. C ++で再帰を使用して、リンクリストの代替ノードを出力します

    リンクリストは、要素を連続していないメモリ位置に格納する線形データ構造です。すべての要素には、リンクリストの次の要素へのポインタが含まれています。 例 − この問題では、リンクリストが与えられ、このリンクリストの要素を印刷する必要がありますが、代替要素のみが印刷されます。問題をよりよく理解するために例を見てみましょう。 Input : 2 -> 4 -> 1 -> 67 -> 48 -> 90 Output : 2 -> 1 -> 48 説明 −リンクリストに代替要素を出力します。したがって、1番目、3番目、5番目の要素が印刷されます。

  2. Pythonの逆リンクリスト

    リンクリストがあるとすると、それを逆にする必要があります。したがって、リストが1→3→5→7のような場合、新しい反転リストは7→5→3→1になります。 これを解決するために、このアプローチに従います- (head、back)を解決するために再帰的な方法でリストの反転を実行する1つの手順を定義します 頭がない場合は、頭を戻します temp:=head.next head.next:=back 戻る=頭 温度が空の場合は、ヘッドを戻します head =temp returnsolve(頭、後ろ) 例 理解を深めるために、次の実装を見てみましょう- class ListNode