6-7 删除循环链表中已知结点的前驱 (10 分)
假设有一个循环链表的长度大于1,且表中既无头节点也无头指针。已知s为指向链表中某结点的指针,试编写算法,在链表中删除指针s所指结点的前驱结点。
函数接口定义:
void DelPrior (Node * s );
其中 s 是传入的参数,函数要求删除s所指结点的前驱结点。其中类型定义如下:
typedef int DataType;
typedef struct node
{
DataType data;
struct node *next;
}LNode,*LinkList;
裁判测试程序样例:
#include <stdio.h>
#include <stdlib.h>
typedef int DataType;
typedef struct node
{
DataType data;
struct node *next;
}LNode,*LinkList;
LinkList Creat_CircularList(); //创建无头结点循环单链表,并返回头指针。
LNode * SetPointerS(LinkList H,int x);//返回指向第x个结点的指针。
void PrintCircularList(LNode *S);//从指针S所指结点开始,依次输出各个结点的值。
void DelPrior(LNode *S);
main()
{
LNode *s;
LinkList H;
int x;
H = Creat_CircularList();
scanf("%d",&x);
s=SetPointerS(H,x);
DelPrior(s);
PrintCircularList(s);
}
LinkList Creat_CircularList()
{
int n,i;
LNode *nw,*rear=NULL,*head=NULL;
scanf("%d",&n);//接收结点总数
for(i=0;i<n;i++)
{
nw=(LNode*)malloc(sizeof(LNode));
scanf("%d",&nw->data);
if(rear==NULL)
rear=head=nw;
else
{
rear->next=nw;
rear=nw;
}
}
if(rear)
rear->next=head;
return head;
}
void PrintCircularList(LinkList s)
{
LNode *p;
for(p=s;p->next!=s;p=p->next)
printf("%d ",p->data);
printf("%d",p->data);
}
LNode * SetPointerS(LinkList H,int x)
{
int i;
LNode *s=H;
for(i=1;i<x;i++)
s=s->next;
return s;
}
/* 请在这里填写答案 */
输入样例:
在这里给出一组输入。例如:
5
1 2 3 4 5
4
输出样例:
在这里给出相应的输出。例如:
4 5 1 2
void DelPrior(LNode *s){
int a;
LNode *p,*q;//q为要删除结点,即s的前驱;p为q的前驱结点
p=q=s;
while(q->next!=s){
q=q->next;
}
while(p->next!=q)
p=p->next;
a=q->data;
p->next=s;
free(q);
}
算法思路:设置两个p,q结点将一个结点q移动到s指针所指到的位置之前,再将p指针移动到q指针之前将q指针所指的内容删除,并释放内存。