约瑟夫问题算法与大数据结构课程设计.docx
- 文档编号:25498841
- 上传时间:2023-06-09
- 格式:DOCX
- 页数:20
- 大小:164.33KB
约瑟夫问题算法与大数据结构课程设计.docx
《约瑟夫问题算法与大数据结构课程设计.docx》由会员分享,可在线阅读,更多相关《约瑟夫问题算法与大数据结构课程设计.docx(20页珍藏版)》请在冰豆网上搜索。
约瑟夫问题算法与大数据结构课程设计
线性表的操作及其应用
一、问题描述
线性表、队列是一种常用的数据结构,有顺序和链式两种存储结构,在实际中应用十分广泛,而链表又分为单链表和循环链表,队列又分为链式队列和循环队列。
这些数据结构都可用来解决约瑟夫环问题。
约瑟夫环问题是算法设计中的一个经典问题,是顺序编号的一组n个人围坐一圈,从第1个人按一定方向顺序报数,在报到m时该人出列,然后按相同方法继续报数,直到所有人出列。
设计算法求约瑟夫环中人员的出列顺序。
二、基本要求
1、选择合适的存储结构,建立线性表;
2、利用顺序存储结构求解约瑟夫环问题;
3、利用单链表和循环链表分别求解约瑟夫环问题;
4、利用队列求解约瑟夫环问题。
三、测试数据
约瑟夫环的测试数据为7,报数为1至3。
四、算法思想
由于用到四种不同的存储结构,它们的算法思想依次是:
1、首先建立一个顺序表模拟整个约瑟夫环,手动输入顺序表长(即参加约瑟夫循环的人数)和循环的次数和表元素。
用已经输出总人数和顺序表长作比较,作为外层循环条件。
并对每一个输出后的元素重新赋值以为标记。
对于每次循环,首先检查顺序表此次是不是我们设立的标记,如果不是则循环次数加1,当达到要求的循环次数时就将循环次数设置为0,输出该元素到屏幕并将总输出元素加1。
每次外循环我们都会移到表的下一个位置,作为新的判断条件,每次报到表尾的时候,我们都将重新设置到表尾,作为下次循环的表元素。
2、首先采用链式循环链表建立整个约瑟夫环,手动输入第一次的循环次数和每个人所持下一个循环次数。
设立判断指针指向表头,并将该指针是否为空作为外层循环条件。
做一个内层循环,将判断指针移动到循环要输出的数,并设立一个前指针指向该指针的前一个位置,输出该元素后,将循环次数重新赋值成该元素。
接着判断前指针和判断指针比较,如果相等说明整个表已经输出完毕,否则将删除该位置的元素。
3、用链式队列建立循环约瑟夫环,手动输入人数,第一次的循环次数和每个人所持下一个循环次数。
并将每一个元素依次入队列,根据第一次循环次数,建立一个for循环,每一次循环都出队列,如果达到要求的循环次数就输出,否则进队列,这样这个数字就出现在队尾。
第一个数输出后,以队列的非空作为循环条件,判断方式如上。
4、用循环队列建立约瑟夫环,将1-7个元素依次进入循环队列,以队列的长度作为与已输出的元素作为判断条件,对每一个输出后的元素重新赋值以为标记。
对于每次循环,首先检查该该位置的元素是不是我们设立的标记-1,如果不是则循环次数加1,将队首指针移到队列的下一个元素,结束此次循环,当达到要求的循环次数时就将重新循环次数设置为0,输出该元素到屏幕并将总输出元素加1。
五、模块划分
1、voidInitQueue(SqQueue*Q)
初始化循环队列
2、voidDestroyQueue(SqQueue*Q)
销毁循环队列
3、voidClearQueue(SqQueue*Q)
清空循环队列
4、intQueueEmpty(SqQueueQ)
判断空队列
5、intQueueLength(SqQueueQ)
求循环队列长度
6、voidGetHead(SqQueueQ,ElemType*e)
取队头元素
7、intEnQueue(SqQueue*Q,ElemTypee)
入队列
8、intDeQueue(SqQueue*Q,ElemType*e)
出队列
9、voidQueueTraverse(SqQueueQ)
遍历循环队列并输出元素
10、voidInitQueue(LQueue*Q)
初始化队列
11、voidDestroyQueue(LQueue*Q)
销毁队列
12、voidClearQueue(LQueue*Q)
清空队列
13、intQueueEmpty(LQueueQ)
判断空队列
14、intQueueLength(LQueueQ)
求队列长度
15、ElemTypeGetHead(LQueueQ,ElemType*e)
取队头元素
16、voidEnQueue(LQueue*Q,ElemTypee)
入队列
17、voidDeQueue(LQueue*Q,ElemType*e)
出队列
18、voidQueueTraverse(LQueueQ)
遍历队列并输出元素
19、voidjoseffer(intn)
约瑟夫环
20、intCreateList(LinkList&L,intn)
建立顺序链表
21、voidjosephus_clist(LinkList&L,intm)
顺序链表解决约瑟夫环
22、voidInitList(SqList*L)
初始化链表
23、voidysf1()
链表解决约瑟夫环
24、voidysf2()
循环链表解决约瑟夫环
25、voidysf3(){
链式队列解决约瑟夫环问题
26、voidysf4()
循环队列解决约瑟夫环问题
27、voidmenu()
菜单
28、intmain()
主函数
六、数据结构//(ADT)
typedefintElemType;
/*定义顺序表类型*/
typedefstruct
{ElemType*elem;
intlength;
}SqList;
/*定义循环链表类型*/
typedefstructLNode
{intnum;
intcode;
structLNode*next;
}*LinkList;
/*定义循环队列类型*/
typedefstruct
{ElemType*base;
intfront;
intrear;
}SqQueue;
/*定义链式队列类型*/
typedefstructQNode
{ElemTypedata;
structQNode*next;
}QNode,*QueuePtr;
typedefstruct
{QueuePtrfront;
QueuePtrrear;
}LQueue;
七、源程序
#include"stdlib.h"
#include"stdio.h"
#defineN100
typedefintElemType;
/*定义顺序表类型*/
typedefstruct
{ElemType*elem;
intlength;
}SqList;
/*定义循环链表类型*/
typedefstructLNode
{intnum;
intcode;
structLNode*next;
}*LinkList;
/*定义循环队列类型*/
typedefstruct
{ElemType*base;
intfront;
intrear;
}SqQueue;
/*定义链式队列类型*/
typedefstructQNode
{ElemTypedata;
StructQNode*next;
}QNode,*QueuePtr;
typedefstruct
{QueuePtrfront;
QueuePtrrear;
}LQueue;
/*初始化循环队列*/
voidInitQueue(SqQueue*Q)
{Q->base=(ElemType*)malloc(N*sizeof(ElemType));
Q->front=Q->rear=0;}
/*销毁循环队列*/
voidDestroyQueue(SqQueue*Q)
{free(Q->base);}
/*清空循环队列*/
voidClearQueue(SqQueue*Q)
{Q->front=Q->rear=0;}
/*判断空队列*/
intQueueEmpty(SqQueueQ)
{if(Q.front==Q.rear)
return1;
else
return0;}
/*求循环队列长度*/
intQueueLength(SqQueueQ)
{return(Q.rear+N-Q.front)%N;}
/*取队头元素*/
voidGetHead(SqQueueQ,ElemType*e)
{if(Q.front!
=Q.rear)
*e=Q.base[Q.front];
}
/*入队列*/
intEnQueue(SqQueue*Q,ElemTypee)
{if((Q->rear+1)%N==Q->front)
return0;
Q->base[Q->rear]=e;
Q->rear=(Q->rear+1)%N;
return1;}
/*出队列*/
intDeQueue(SqQueue*Q,ElemType*e)
{if(Q->front==Q->rear)
return0;
*e=Q->base[Q->front];
Q->front=(Q->front+1)%N;
return1;}
/*遍历循环队列并输出元素*/
voidQueueTraverse(SqQueueQ)
{inti;
printf("\nQueue:
");
if(Q.rear for(i=Q.front;i printf("%d\t",Q.base[i%N]); } /*初始化队列*/ voidInitQueue(LQueue*Q) {Q->front=Q->rear=(QueuePtr)malloc(N*sizeof(QNode)); if(! (Q->front))exit(0); Q->front->next=NULL; } /*销毁队列*/ voidDestroyQueue(LQueue*Q) {while(Q->front) {Q->rear=Q->front->next; free(Q->front); Q->front=Q->rear;} } /*清空队列*/ voidClearQueue(LQueue*Q) {QueuePtrp; p=Q->front->next; while(p) {Q->front->next=p->next; free(p); Q->rear=Q->front;} } /*判断空队列*/ intQueueEmpty(LQueueQ) {if(Q.front==Q.rear) return1; else return0;} /*求队列长度*/ intQueueLength(LQueueQ) {QueuePtrp;intn=0; p=Q.front; while(p! =Q.rear) {n++;p=p->next;} returnn; } /*取队头元素*/ ElemTypeGetHead(LQueueQ,ElemType*e) {if(Q.front! =Q.rear) returnQ.front->next->data; elsereturn0; } /*入队列*/ voidEnQueue(LQueue*Q,ElemTypee) {QueuePtrp; p=(QueuePtr)malloc(N*sizeof(QNode)); if(! p)exit(0); p->data=e; p->next=NULL; Q->rear->next=p; Q->rear=p; } /*出队列*/ voidDeQueue(LQueue*Q,ElemType*e) {QueuePtrp; if(Q->front! =Q->rear) {p=Q->front->next; *e=p->data; Q->front->next=p->next; if(Q->rear==p) Q->rear=Q->front; free(p); } } /*遍历队列并输出元素*/ voidQueueTraverse(LQueueQ) {QueuePtrp; printf("\nQueue: "); p=Q.front->next; while(p) {printf("%d\t",p->data); p=p->next; } } /*约瑟夫环*/ voidjoseffer(intn) {LQueueQ; inti; ElemTypex; InitQueue(&Q); for(i=1;i<=n;i++) EnQueue(&Q,i); while(! QueueEmpty(Q)) {for(i=1;i<=3;i++) {DeQueue(&Q,&x); if(i! =3)EnQueue(&Q,x); elseprintf("%4d",x); } } } /*建立顺序链表*/ intCreateList(LinkList&L,intn) {LNode*p,*q; printf("Input%dnumber: \n",n); L=q=(LinkList)malloc(sizeof(LNode)); if(L==NULL||q==NULL) return0; for(inti=1;i<=n;i++) {p=(LinkList)malloc(sizeof(LNode)); scanf("%d",&p->code); p->num=i; if(i==1) L=p; else q->next=p; q=p; } p->next=L; L=p; return1; } /*顺序链表解决约瑟夫环*/ voidjosephus_clist(LinkList&L,intm) {inte=m,k; LinkListp,pre,tp; p=L; while(p! =NULL) {for(intj=0;j {pre=p; p=p->next;} e=p->code; k=p->num; printf("Theoutputnumberis%d\n",e); if(pre! =p) {tp=p; pre->next=p->next; p=p->next; p=pre; free(tp); } else {free(pre); p=NULL;} } } /*初始化链表*/ voidInitList(SqList*L) {L->elem=(ElemType*)malloc(N*sizeof(ElemType)); L->length=0;} /*链表解决约瑟夫环*/ voidysf1() {SqListL;intm,i,d,k; InitList(&L); printf("元素个数和报到第几出"); scanf("%d%d",&L.length,&d); printf("输入元素"); for(i=0;i scanf("%d",&L.elem[i]); m=0;k=0;i=0; while(m {if(L.elem[i]! =N)k++; if(k==d) {printf("%3d",L.elem[i]); L.elem[i]=N;k=0;m++;} i++; if(i==L.length)i=0; } } /*循环链表解决约瑟夫环*/ voidysf2() {intm,n; LinkListjos_clist; printf("Inputthebegingingnumber: \n"); scanf("%d",&m); printf("Inputnpeople: \n"); scanf("%d",&n); CreateList(jos_clist,n); josephus_clist(jos_clist,m); } /*链式队列解决约瑟夫环问题*/ voidysf3() {LQueueQ; inti;ElemTypex;intn; printf("number: n="); scanf("%d",&n); InitQueue(&Q); for(i=1;i<=n;i++) EnQueue(&Q,i); printf("len: %d\n",QueueLength(Q)); while(! QueueEmpty(Q)) {DeQueue(&Q,&x); printf("%d\t",x);} QueueTraverse(Q); joseffer(n); } /*循环队列解决约瑟夫环问题*/ voidysf4() {SqQueueQ;inti,m,k,j; k=0;m=0; InitQueue(&Q); for(i=1;i<=7;i++) EnQueue(&Q,i); j=QueueLength(Q); printf("len=%d",j); QueueTraverse(Q); printf("\nJoseffer: \t"); while(m<=QueueLength(Q)) {if(Q.base[Q.front]! =-1) {k++;} if(k==3) {printf("%d\t",Q.base[Q.front]); Q.base[Q.front]=-1; k=0; m++;} Q.front=(Q.front+1)%j;} } /*菜单*/ voidmenu() {intchoice; do{ printf("\n===================================="); printf("\n主菜单"); printf("\n1.顺序表实现约瑟夫环问题"); printf("\n2.循环链表表实现约瑟夫环问题"); printf("\n3.链式队列实现约瑟夫环问题"); printf("\n4.循环队列实现约瑟夫环问题"); printf("\n5.退出程序"); printf("\n===================================="); printf("\n请输入您的选择(1,2,3,4,5)choice="); scanf("%d",&choice); switch(choice) {case1: ysf1();break; case2: ysf2();break; case3: ysf3();break; case4: ysf4();break; case5: exit(0);} }while(choice<=5); } /*主函数*/ intmain() {menu(); System("pause"); return0; } 八、测试情况 程序的测试结果如下: 1、顺序表实现约瑟夫环问题 测试结果正确。 2、循环链表实现约瑟夫环问题 测试结果正确。 3、链式队列实现约瑟夫环问题 测试结果正确。 4、链式队列实现约瑟夫环问题 测试结果正确。 九、参考文献 1、严蔚敏,《数据结构C语言》,清华大学出版社。 2、谭浩强,《c语言程序设计》,清华大学出版社。 小结 约瑟夫环问题讲的是: n个人围成一圈报数,报m的人自动离开,然后下位重新从1开始报数,一直循环直至结束。 本次课程设计是对于约瑟夫环基于不同存储方式的实现。 首先我们对此算法可以进行伪代码的语言描述,这样可以在以后的设计中有清晰的思路,从而不引起在写代码时期的混乱。 具体实现期间我们可以通过“程序=算法+数据”的公式,知道我们首先该考虑的是约瑟夫的数据模型,在这里表现的就是C语言的结构体变量,有了数据模型我们就可以在上面进行各种各样的操作,结构体成员变量的思考是本次设计很重要的一部分。 至于约瑟夫环的算法可以通过上面先请描述的伪代码一步一步换成函数或者程序语言之类。 程序编写好后,在排除了编译错误之后,如果出现运行结果的不对,这就可以通过VC的Debug模式下进行单步调试,通过按F10一步一步的执行,分析出错代码的位置,然后进行排除。 一般的可能数组越界或者指针之类的问题最为突出。 通过本次课程设计加强了我们对于C语言的编写能力,锻炼了我们编写一个小型项目的能力,在以后遇到类似的项目时有了清晰的解决思路,也体会到了团队合作的力量。 我们也学到了很多课内学不到的东西,比如独立思考解决问题,出现差错的随机应变,和与人合作共同提高,都受益非浅,今后的应该更轻松,自己也都能扛的起并高质量的完成项目。 同时我们发现考试并不是最重要的,只有理论知识是远远不够的,能运用所学的理论知识解决遇到的实际问题才是最重要的,只有把所学的理论知识与实践相结合起来,从理论中得出结论,才能真正为社会服务,从而提高自己的实际动手能力和独立思考的能力。 本次设计的过程中遇到很多问题,可以说得是困难重重,这毕竟第一次做的,难免会遇到过各种各样的问题,同时在设计的过程中发现了自己的不足之处,对以前所学过的知识理解得不够深刻,掌握得不够牢固。 最后,我们真正体会到什么是团队协作,真正的了解到团队合作的有利之处,在我们完成课程设计过程中学会积极的同团队成员交流,取长补短,共同进步。 只有和同学多交流多学习才能不断的提高自身水平真正感受到团队成员为了共同的目标联合在一起时的强大力量。 最后要感谢在课程设计过程中给予帮助的老师和同学!
- 配套讲稿:
如PPT文件的首页显示word图标,表示该PPT已包含配套word讲稿。双击word图标可打开word文档。
- 特殊限制:
部分文档作品中含有的国旗、国徽等图片,仅作为作品整体效果示例展示,禁止商用。设计者仅对作品中独创性部分享有著作权。
- 关 键 词:
- 约瑟夫 问题 算法 数据结构 课程设计