欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

C_线性表(ADT)-单向循环链表的表示和实现

程序员文章站 2022-08-04 18:11:12
循环链表是另一种形式的链式存贮结构。它的特点是表中最后一个结点的指针域指向头结点,整个链表形成一个环。表中最后一个结点的指针域指向头结点,整个链表形成一个环。嗯下面就网上找的一张单链的循环链表 单...

循环链表是另一种形式的链式存贮结构。它的特点是表中最后一个结点的指针域指向头结点,整个链表形成一个环。表中最后一个结点的指针域指向头结点,整个链表形成一个环。嗯下面就网上找的一张单链的循环链表

C_线性表(ADT)-单向循环链表的表示和实现

单项链表定义:

typedef struct lnode{
	elemtype data;
	struct lnode *next;	
}lnode,*linklist;

单向循环链表基本操作:

 

/*操作结果:构造一个空的线性表l*/
void initlist(linklist &l)

/*初始条件:单循环链表l存在*/
/*操作结果:摧毁单循环链表l*/
void destroylist(linklist &l)

/*初始条件:单循环链表l存在*/
/*操作结果:将l重置为空表*/
status clearlist(linklist &l)

/*初始条件:单循环链表l存在*/
/*操作结果:若l为空表,则返回true,否则返回false*/
status listempty(linklist l)

/*初始条件:单循环链表l存在*/
/*操作结果:返回l中数据元素个数*/ 
int listlength(linklist l)

/*初始条件:单循环链表l存在*/
/*操作结果:当第i个元素存在时,其值赋给e并返回ok,否则返回false。*/
status getelem(linklist l,int i,elemtype &e)

/*初始条件:单循环链表l存在,compare()是数据元素判定的函数*/
/*操作结果:返回l中第1个与e满足关系compare()的数据元素的位序。若这样的数据元素不存在,则返回值为error*/
status locateelem(linklist l,elemtype e,status(*compare)(elemtype,elemtype))

/*初始条件:单循环链表l存在*/
/*操作结果:若cur_e是l的数据元素,且不是第一个,则用pre_e返回它的前驱;否则,操作失败,pre_e无定义*/
status priorelem(linklist l,elemtype cur_e,elemtype &pre_e)


/*初始条件:单循环链表l存在*/
/*操作结果:若cur_e是l的数据元素,且不是最后一个,则用next_e返回它的后继;否则,操作失败,next_e无定义*/
status nextelem(linklist l,elemtype cur_e,elemtype &next_e)

/*初始条件:单循环链表l存在*/
/*操作结果:在l的第i个位置之前插入元素e*/
status listinsert(linklist &l,int i,elemtype e) 

/*初始条件:单循环链表l存在*/
/*操作结果:删除l的第i个元素,并由e返回其值*/
status listdelete(linklist &l,int i,elemtype &e)

/*初始条件:单循环链表l存在*/
/*操作结果:依次对每个数据元素调用函数visit()*/
void listtraverse(linklist l,void(*visit)(elemtype))


 

单向链表操作基本实现:

构造一个空的线性表l

 

/*操作结果:构造一个空的线性表l*/
void initlist(linklist &l)
{
	l=(linklist)malloc(sizeof(lnode));		//产生头结点,并使l指向此头结点 
	if(!l){//空的线性表构造失败
		printf("空的线性表构造失败,退出程序!\n");
		exit(0); 
		}
	l->next=l;		//指针域指向头结点 
}


 

摧毁单循环链表l

 

/*初始条件:单循环链表l存在*/
/*操作结果:摧毁单循环链表l*/
void destroylist(linklist &l)
{
	linklist q;
	linklist p=l->next;		//p指向头结点
	while(p!=l)				//没到表尾
	{
		q=p->next;
		free(p);
		p=q;
	}
	free(l);
	l=null;
}


 

将l重置为空表

 

/*初始条件:单循环链表l存在*/
/*操作结果:将l重置为空表*/
status clearlist(linklist &l)
{
	linklist p,q;
	l=l->next;			//l指向头结点
	p=l->next;			//p指向第一个结点
	while(p!=l)			//没到表尾
	{
		q=p->next;
		free(p);
		p=q;
	}
	l->next=l;			//头结点指针域指向自身 
}

对线性表进行判空

 

 

/*初始条件:单循环链表l存在*/
/*操作结果:若l为空表,则返回true,否则返回false*/
status listempty(linklist l)
{
	if(l->next==l)		//线性表为空表
		return true;
	else
		return false; 
}

返回表长

 

 

/*初始条件:单循环链表l存在*/
/*操作结果:返回l中数据元素个数*/ 
int listlength(linklist l)
{
	int i=0;
	linklist p=l->next;		//p指向头结点
	while(p!=l)				//没到表尾
	{
		i++;
		p=p->next;
	} 
	return i;
} 

求链表位序结点

 

 

/*初始条件:单循环链表l存在*/
/*操作结果:当第i个元素存在时,其值赋给e并返回ok,否则返回false。*/
status getelem(linklist l,int i,elemtype &e)
{
	int count=1;
	linklist p=l->next->next;		//p指向第一个结点
	if(i<=0||i>listlength(l))		//第i个元素不存在时
		return error;
	while(countnext;
		count++;	
	}
	e=p->data;						//取第i个元素
	return ok; 
}

求链表结点位序

 

 

/*初始条件:单循环链表l存在,compare()是数据元素判定的函数*/
/*操作结果:返回l中第1个与e满足关系compare()的数据元素的位序。
			若这样的数据元素不存在,则返回值为error*/
status locateelem(linklist l,elemtype e,status(*compare)(elemtype,elemtype))
{
	int i=0;
	linklist p=l->next->next;		//p指向第一个结点
	while(p!=l->next)
	 {
 		i++;
 		if(compare(p->data,e))		//满足关系
		 		return i;
		p=p->next; 
 	}
 	return 0;
}

求前驱结点

 

 

/*初始条件:单循环链表l存在*/
/*操作结果:若cur_e是l的数据元素,且不是第一个,则用pre_e返回它的前驱;
			否则,操作失败,pre_e无定义*/
status priorelem(linklist l,elemtype cur_e,elemtype &pre_e)
{
	linklist q;
	linklist p=l->next->next;			//p指向第一个结点
	q=p->next;
	while(q!=l->next)					//p没到表尾
	{
		if(q->data==cur_e)
		{
			pre_e=p->data;
			return true;
		}
		p=q;
		q=q->next;
	}
	return false;		//操作失败	
}

求结点后继

 

 

/*初始条件:单循环链表l存在*/
/*操作结果:若cur_e是l的数据元素,且不是最后一个,则用next_e返回它的后继;
			否则,操作失败,next_e无定义*/
status nextelem(linklist l,elemtype cur_e,elemtype &next_e)
{
	linklist p=l->next->next;		//p指向第一个结点
	while(p!=l)						//p没有到达链表尾 
	{
		if(p->data==cur_e)
		{
			next_e=p->next->data;
			return true;
		}
		p=p->next;
	}
	return false;					//操作失败	
}

链表中插入新结点

 

 

/*初始条件:单循环链表l存在*/
/*操作结果:在l的第i个位置之前插入元素e*/
status listinsert(linklist &l,int i,elemtype e) 
{
	linklist s;
	linklist p=l->next;				//p指向头结点
	int count=0;
	if(i<0||i>listlength(l)+1)		//无法在第i个元素前插入
		return error;
	while(countnext;
		count++;	
	}
	s=(linklist)malloc(sizeof(lnode));		//生成新结点
	s->data=e;								//插入l中
	s->next=p->next;
	p->next=s;
	if(p==l)							//改变尾结点
		l=s;
	return ok; 
} 

链表中删除结点

 

 

/*初始条件:单循环链表l存在*/
/*操作结果:删除l的第i个元素,并由e返回其值*/
status listdelete(linklist &l,int i,elemtype &e)
{
	linklist p=l->next,q;			//p指向头结点
	int count=0;
	if(i<=0||i>listlength(l))		//第i个元素不存在
		return error;
	while(countnext;
		count++;
	}
	q=p->next;						//q指向删除结点
	p->next=q->next;
 	e=q->data;
    if(l==q)						//删除的是表尾元素
	 	l=p;
	free(q);
	return ok; 
}

依次查看结点元素

 

 

/*初始条件:单循环链表l存在*/
/*操作结果:依次对每个数据元素调用函数visit()*/
void listtraverse(linklist l,void(*visit)(elemtype))
{
	linklist p=l->next->next;		//p指向第一个结点
	while(p!=l->next)				//p不指向头结点
	{
		visit(p->data);
		p=p->next;
	}
	printf("\n");
} 


 

嗯下面就是在vc中的测试

 

//单循环链表 
#include
#include
#include
#define ok 1
#define error 0
#define true 1
#define false 0

typedef int elemtype;
typedef int status;
typedef struct lnode{
	elemtype data;
	struct lnode *next;	
}lnode,*linklist;

status compare(elemtype e1,elemtype e2)
{
	if(e1==e2)
		return 0;
	else if(e1>e2)
		return 1;
	else if(e1<e2) return="" -1;="" }="" status="" equal(elemtype="" c1,elemtype="" c2)="" {="" if(c1="=c2)" true;="" else="" error;="" void="" visit(elemtype="" e)="" printf("%d="" \n",e);="" print(elemtype="" c)="" ",c);="" *操作结果:构造一个空的线性表l*="" initlist(linklist="" &l)="" l="(linklist)malloc(sizeof(lnode));" 产生头结点,并使l指向此头结点="" if(!l){="" 空的线性表构造失败="" printf("空的线性表构造失败,退出程序!\n");="" exit(0);="" l-="">next=l;		//指针域指向头结点 
}

/*初始条件:单循环链表l存在*/
/*操作结果:摧毁单循环链表l*/
void destroylist(linklist &l)
{
	linklist q;
	linklist p=l->next;		//p指向头结点
	while(p!=l)				//没到表尾
	{
		q=p->next;
		free(p);
		p=q;
	}
	free(l);
	l=null;
}

/*初始条件:单循环链表l存在*/
/*操作结果:将l重置为空表*/
status clearlist(linklist &l)
{
	linklist p,q;
	l=l->next;			//l指向头结点
	p=l->next;			//p指向第一个结点
	while(p!=l)			//没到表尾
	{
		q=p->next;
		free(p);
		p=q;
	}
	l->next=l;			//头结点指针域指向自身 
}

/*初始条件:单循环链表l存在*/
/*操作结果:若l为空表,则返回true,否则返回false*/
status listempty(linklist l)
{
	if(l->next==l)		//线性表为空表
		return true;
	else
		return false; 
}

/*初始条件:单循环链表l存在*/
/*操作结果:返回l中数据元素个数*/ 
int listlength(linklist l)
{
	int i=0;
	linklist p=l->next;		//p指向头结点
	while(p!=l)				//没到表尾
	{
		i++;
		p=p->next;
	} 
	return i;
} 

/*初始条件:单循环链表l存在*/
/*操作结果:当第i个元素存在时,其值赋给e并返回ok,否则返回false。*/
status getelem(linklist l,int i,elemtype &e)
{
	int count=1;
	linklist p=l->next->next;		//p指向第一个结点
	if(i<=0||i>listlength(l))		//第i个元素不存在时
		return error;
	while(countnext;
		count++;	
	}
	e=p->data;						//取第i个元素
	return ok; 
}

/*初始条件:单循环链表l存在,compare()是数据元素判定的函数*/
/*操作结果:返回l中第1个与e满足关系compare()的数据元素的位序。
			若这样的数据元素不存在,则返回值为error*/
status locateelem(linklist l,elemtype e,status(*compare)(elemtype,elemtype))
{
	int i=0;
	linklist p=l->next->next;		//p指向第一个结点
	while(p!=l->next)
	 {
 		i++;
 		if(compare(p->data,e))		//满足关系
		 		return i;
		p=p->next; 
 	}
 	return 0;
}

/*初始条件:单循环链表l存在*/
/*操作结果:若cur_e是l的数据元素,且不是第一个,则用pre_e返回它的前驱;
			否则,操作失败,pre_e无定义*/
status priorelem(linklist l,elemtype cur_e,elemtype &pre_e)
{
	linklist q;
	linklist p=l->next->next;			//p指向第一个结点
	q=p->next;
	while(q!=l->next)					//p没到表尾
	{
		if(q->data==cur_e)
		{
			pre_e=p->data;
			return true;
		}
		p=q;
		q=q->next;
	}
	return false;		//操作失败	
}

/*初始条件:单循环链表l存在*/
/*操作结果:若cur_e是l的数据元素,且不是最后一个,则用next_e返回它的后继;
			否则,操作失败,next_e无定义*/
status nextelem(linklist l,elemtype cur_e,elemtype &next_e)
{
	linklist p=l->next->next;		//p指向第一个结点
	while(p!=l)						//p没有到达链表尾 
	{
		if(p->data==cur_e)
		{
			next_e=p->next->data;
			return true;
		}
		p=p->next;
	}
	return false;					//操作失败	
}

/*初始条件:单循环链表l存在*/
/*操作结果:在l的第i个位置之前插入元素e*/
status listinsert(linklist &l,int i,elemtype e) 
{
	linklist s;
	linklist p=l->next;				//p指向头结点
	int count=0;
	if(i<0||i>listlength(l)+1)		//无法在第i个元素前插入
		return error;
	while(countnext;
		count++;	
	}
	s=(linklist)malloc(sizeof(lnode));		//生成新结点
	s->data=e;								//插入l中
	s->next=p->next;
	p->next=s;
	if(p==l)							//改变尾结点
		l=s;
	return ok; 
} 

/*初始条件:单循环链表l存在*/
/*操作结果:删除l的第i个元素,并由e返回其值*/
status listdelete(linklist &l,int i,elemtype &e)
{
	linklist p=l->next,q;			//p指向头结点
	int count=0;
	if(i<=0||i>listlength(l))		//第i个元素不存在
		return error;
	while(countnext;
		count++;
	}
	q=p->next;						//q指向删除结点
	p->next=q->next;
 	e=q->data;
    if(l==q)						//删除的是表尾元素
	 	l=p;
	free(q);
	return ok; 
}

/*初始条件:单循环链表l存在*/
/*操作结果:依次对每个数据元素调用函数visit()*/
void listtraverse(linklist l,void(*visit)(elemtype))
{
	linklist p=l->next->next;		//p指向第一个结点
	while(p!=l->next)				//p不指向头结点
	{
		visit(p->data);
		p=p->next;
	}
	printf("\n");
} 


int main()
{
	int n;
	int ch;
	linklist l;
	elemtype e;
	initlist(l);		//初始化单循环链表
	printf("单循环链表构造成功!\n");
	printf("请输入单循环链表的长度:");
	scanf("%d",&n);
	printf("请输入单循环链表各个数据:");
	for(int i=1;i<=n;i++)
	{
		scanf("%d",&e);
		listinsert(l,i,e);			//在第i个结点前插入e 
	} 
	printf("*********************************\n");
	printf("1、插入结点\n2、删除结点\n3、链表判空\n");
	printf("4、链表长度\n5、清空链表\n6、销毁链表\n");
	printf("7、结点前驱\n8、结点后续\n9、遍历链表\n");
	printf("10、查询位序结点\n");
	printf("11、查询结点位序\n");
	printf("0、退出操作\n"); 
	printf("*********************************\n");
	printf("请选择接下来要进行的操作:");
	while(scanf("%d",&ch)&&ch!=0)
	{
		if(ch==1){
			int set;
			printf("请输入在第几个结点之前插入新结点:");
			scanf("%d",&set);
			printf("请输入要插入结点的数据:");
			scanf("%d",&e);
			if(listinsert(l,set,e))
						printf("操作成功!\n");
		}
		if(ch==2){
			int set;
			printf("请输入要删除第几个结点:");
			scanf("%d",&set);
			if(listdelete(l,set,e))
				printf("成功删除第%d个结点,其值为%d\n",set,e);
		}
		if(ch==3){
			if(listempty(l))
				printf("这是一个空的链表!\n");
			else
				printf("这不是一个空的链表!\n");
		}
		if(ch==4){
			printf("链表的长度为:%d\n",listlength(l));
		}
		if(ch==5){
			if(clearlist(l))
				printf("已成功清空链表!\n");
		}
		if(ch==6){
			destroylist(l);
		}
		if(ch==7){
			printf("请输入要查找元素的前驱:");
			scanf("%d",&n);
			if(priorelem(l,n,e))
				printf("%d元素的前驱是%d\n",n,e);
			else
				printf("输入错误!\n");
		}
		if(ch==8){
			printf("请输入要查找元素的后续:");
			scanf("%d",&n);
			if(nextelem(l,n,e))
				printf("%d元素的后续是%d\n",n,e);
			else
				printf("输入错误!\n");//输入的是尾结点也会判断尾输入错误 
		}
		if(ch==9){
			printf("正序输出链表:");
			listtraverse(l,print);
		}
		if(ch==10){
			printf("请输入要匹配的数据元素:");
			scanf("%d",&n);
		 	if(locateelem(l,n,equal))
		 		printf("等于%d的元素是第%d个\n",n,locateelem(l,n,equal));
		 	else
	 			printf("链表中不存在元素%d\n",n);
		}
		if(ch==11){
			int set;
			printf("请输入要查询链表位序的值:");
			scanf("%d",&set);
			if(getelem(l,set,e))
				printf("链表的第%d个元素的值为%d\n",set,e);
			else
				printf("输入错误!\n");
		}
	printf("请选择接下来要进行的操作:");
	}
	printf("成功退出操作!\n");
	return 0;
}