3-5 单链表分段逆转 (20分)
给定一个带头结点的单链表和一个整数K,要求你将链表中的每K个结点做一次逆转。例如给定单链表 1→2→3→4→5→6 和 K=3,你需要将链表改造成 3→2→1→6→5→4;如果 K=4,则应该得到 4→3→2→1→5→6。
函数接口定义:
void K_Reverse( List L, int K );
其中List
结构定义如下:
typedef struct Node *PtrToNode;
struct Node {
ElementType Data; /* 存储结点数据 */
PtrToNode Next; /* 指向下一个结点的指针 */
};
typedef PtrToNode List; /* 定义单链表类型 */
L
是给定的带头结点的单链表,K
是每段的长度。函数K_Reverse
应将L
中的结点按要求分段逆转。
裁判测试程序样例:
#include <stdio.h>
typedef int ElementType;
typedef struct Node *PtrToNode;
struct Node {
ElementType Data; /* 存储结点数据 */
PtrToNode Next; /* 指向下一个结点的指针 */
};
typedef PtrToNode List; /* 定义单链表类型 */
List ReadInput(); /* 裁判实现,细节不表 */
void PrintList( List L ); /* 裁判实现,细节不表 */
void K_Reverse( List L, int K );
int main()
{
List L;
int K;
L = ReadInput();
scanf("%d", &K);
K_Reverse( L, K );
PrintList( L );
return 0;
}
/* 你的代码将被嵌在这里 */
输入样例:
6
1 2 3 4 5 6
4
输出样例:
4 3 2 1 5 6
思路:
分段逆转链表,无非就是每隔K个结点,将前K个结点按照头插法存储到另一个链表里,再和原链表没逆转的连接起来,照此规律进行循环逆转,直到剩下的未逆转元素个数小于K个,直接连接即可。
AC代码:
void K_Reverse( List L, int K ){
List p=L->Next; //p的作用就是参与记录表长
int count=0;
while(p!=NULL){ //记录表长到count里
count++;
p=p->Next;
}
while(K<=count&&K!=1){ //如果K=1.表不需要逆转
List L1=NULL,L2=NULL,head=L;
int i=1;
L=L->Next;
while(i<=K && L!=NULL){ //利用头插法进行K个数的逆转
L1=(struct Node *)malloc(sizeof(List));
L1->Data=L->Data;
L1->Next=L2;
L2=L1;
L=L->Next;
i++;
}
head->Next=L1;
p=head; //p在这里只起到一个连接的作用
while(p->Next!=NULL) //找到逆转序列L1的最后一个位置
p=p->Next;
p->Next=L; //连接未逆转的链表L
L=p;
count=count-K; //计算未逆转链表元素个数
if(!count)
return;
}
}