车是首节点(首子)
马是次节点(次子)
牙签细的是指针指向,香头发黑的是指向,铁头细的是指向。
根据步骤写程序的伪算法(3步4循环,7张图片搞定),如下:
以下是while循环(条件:香头指向不为空)
第一个循环把马弄到车前面,
第二个循环把相弄到马前面
第三个循环把士弄到相前面
........
直到香指向为空后停止循环。
代码如下:只需要一个首结点pHead,就能把链表找到,并倒置。具体代码如下
p香=pHead->pNext
p铁=p香->pNext
p香->pNext=NULL
P香=p铁
while(p香 !=NULL)
{
p铁=p香->pNext
p香->pNext=pHead->pNext
pHead->pNext=p香;
p香=p铁;
}
对照伪算法(三步四循环),和上面的代码是一一对应的:
第一步:香头指向首子,铁头指向次子
第二步:删掉首子指向次子(铁头所指向的那个子)的牙签
第三步:香头跟着铁头
以下循环条件:(条件:香头指向不为空)
{
循环1:铁头移动到香头的下一个指向
循环2:香头的下一个指向首子
循环3:头子的下一个跟着香头
循环4:香头跟着铁头
}
自己用道具操作几遍,然后把流程背会,以后自己根据流程写代码即可。
算法的核心就是reverse函数,其它的都是辅助建立链表和输出链表的。从数据结构的定义来看这是一个带头节点的链表。要弄的十分明白你要耐心点画图看看。我简单的说下思路:
【head是指向头结点的】
p=head //p最开始指向头结点
s=p->next//s最开始指向第一个节点
while(s->next!=NULL)//只要没有到最后一个元素就继续。最后一个元素的next肯定为NULL
{ //进入核心了楼主
t=s->next //用t指向s后面的那个元素
s->next=p//把s指向她前面那个,这个时候就实现了逆向了。而且是就地逆向。元素都没有动的
p=s//然后p向后移动s
s=t//s向后移动到p
这样到下一轮的时候又可以讲下下个再指向刚才那个下一个。一次内推
}
s->next=p //当最后一个的时候,还是要指向她的前一个。
head->next->next=NULL//头指针的下一个是指向原来的第一个。逆向后肯定是最后的那个了。所以最后的一个的next=NULL就明了了。
head->next=s//s是逆序前的最后一个,逆序后是第一个,所以用头指向他
画个图好好体会下,楼主!
#include<stdio.h>#include<stdlib.h>
#include <malloc.h>
//定义链表节点
typedef struct LNode
{ int data
struct LNode *next
} LNode, *Linklist
//创建链表
Linklist create()
{ int i,n //i用于下面循环,n用来存放有效节点的字数
Linklist p,L
printf("Please input N =")
scanf("%d",&n)
L= (Linklist)malloc(sizeof(LNode)) // 分配一个不存放有效数据的头结点
L->next=NULL
for(i = 0 i < n i++)
{ p = (Linklist)malloc(sizeof(LNode))//生成新节点
scanf("%d",&p->data)//输入元素值
p->next = L->next
L->next = p
}
return L //返回头节点
}
//链表反转输出
Linklist ReverseList(Linklist L,int st) //st为1时输出结点数据
{ if(L->next!= NULL)
ReverseList(L->next,1)
if(st)printf("%d ", L->data)
return L
}
void put(Linklist L)
{ Linklist p
p = L->next
while(p != NULL)
{ printf("%d ", p->data)
p = p->next
}
printf("\n")
}
int main()
{ Linklist L
L=create()
printf("A: ") put(L)
printf("B: ")
ReverseList(L,0) //附加结点未保存数据,故第二参数为0
return 0
}