数据结构课程设计joseph环

上传人:仙*** 文档编号:34617129 上传时间:2021-10-22 格式:DOC 页数:18 大小:5.18MB
收藏 版权申诉 举报 下载
数据结构课程设计joseph环_第1页
第1页 / 共18页
数据结构课程设计joseph环_第2页
第2页 / 共18页
数据结构课程设计joseph环_第3页
第3页 / 共18页
资源描述:

《数据结构课程设计joseph环》由会员分享,可在线阅读,更多相关《数据结构课程设计joseph环(18页珍藏版)》请在装配图网上搜索。

1、Joseph环 1. 课程设计目的 ⑴ 通过独立解决某个课程设计问题,在数据结构的逻辑特性和物理表示、数据结构的选择应用、算法的设计及其实现等方面加深对课程基本内容的理解和综合运用。 ⑵ 深刻理解、牢固掌握数据结构和算法设计技术, 提高分析和解决实际问题的能力。 ⑶ 在程序设计方法以及上机操作等基本技能和科学作风方面进行比较系统和严格的训练。 2. 设计方案论证 2.1 设计思路 首先,定义两个结构体,将个人的信息写入其中内容包括个人的顺序号(Num),个人的密码m (随机输入的值)及指针. 第二,再将每个人的信息存储于一个单向循环链表内. 第三,根据题目要求编写程序,开

2、始随机把一个数赋给m,开始报数(查找)则将顺序号为m的人的编号提出列,并将其的密码(随机输入的)赋给m. 最后,m有了新值,再从出列的人的下一个位置开始重复上面第三步,直到所有人的顺序号都被调出,结束程序。 2.2设计方法: 2.2.1 结构设计 利用单向循环链表存储结构模拟此过程,按照出列的顺序输出各个人的编号。输入数据:建立输入函数处理输入数据,输入m值和n值,存放在由DATA结构体为存储元素的顺序存储结构里。再建立单向循环链表,节点结构为NODE结构体,将所有信息存在链表里。输出形式:建立一个输出函数,将正确的输出序列。 2.3算法设计 2.3.1 线性表的单链表存储

3、结构 typedef struct node {int num; //成员编号 int m; //每个成员的唯一编码 struct node *next; }NODE,*LINK; 2.3.2 线性表的动态分配顺数存储结构 typedef struct data //输入函数中数据存储节点结构 {int num; int m; } DATA; 2.3.3 数据输入函数 DATA InputData() 输入函数采用线性表的顺序存储结构,线性表的顺序存储结构是一种随机存取的存

4、储结构。特点是存储空间连续,用一组地址连续的存储单元依次存储线性表的数据元素。并且逻辑位置相邻的两个元素其物理位置也相邻。每一个数据元素的存储位置都和线性表的起始位置相差一个和数据元素在线性表中的位序成正比的常数,由此,只要确定了存储线性表的起始位置,线性表中的任一数据元素都可以随机存取。由于高级程序设计语言中的数组类型也有随机存取的特性,因此,通常都用数组来描述数据结构中的顺序存储结构。线性表的顺序存储结构需要预先分配存储空间。 输入函数算法设计如下: DATA *InputData (int *n) //数据输入函数,n 是节点的个数 {DATA *d, *p,

5、 *s; //d用来存放数组空间的首地址 int i; printf("How many people there are?\n"); printf("Please Input length:"); scanf("%d",n); // 输入节点数 d=(DATA *)malloc(*n*sizeof(DATA)); //申请内存空间函数malloc返回首地址存放在DATA类型的d变量里 p=d; for(i=1;i<=*n;i++) //动态输入各节点信息 {printf("Number%d ---- input m:

6、 ( m means the secret of them! ) :",i); scanf("%d",&p->m); p->num=i; p++; } return(d); //返回内存空间的首地址 } 输入函数流程图如下: 开始 输入n d=(DATA *)malloc(*n*sizeof(DATA)); p=d ; i=1 i<=*n 输入m p->num=i;p++ return(d) 结束 Y N 图1 输入函数流程图 2.3.4 单向循环链表的建立

7、LINK CreateLinklist ( DATA *d, int n ) 单向循环链表是一种链式存储结构,它的特点是表中最后一个结点的指针域指向头结点,整个链表形成一个环,由此,从表中任意结点出发均可找到表中其他结点。单向循环链表的构造类似于单向链表的构造,但是不同的是最后一个元素的指针指向的是首元素。 循环链表的操作和线性链表基本一致,差别仅在于算法中的循环条件不是p或p->next是否为空,而是他们是否指向头指针,但有的时候,若在循环链表中设置尾指针而不设头指针,即带尾指针的单向循环链表,可使某些程序简化。 单链表和顺序存储结构不同,它是一种动态结构,整个可用存储空间可为多个

8、链表共同享用,每个链表占用的空间不需预先分配划定,而可由系统应需求直接生成,因此,建立线性表线性存储结构的过程就是一个动态生成链表的过程。 在单链表中任何两个元素的存储位置之间没有固定的联系。但每个元素的存储位置都包含在其直接前驱节点的信息之中。假设p是指向线性表中第i个数据元素(节点ai)的指针,则p->next是指向第i+1个数据元素(节点ai+1)的指针。换句话说,若p->data=ai,则p->next->data=ai+1。由此,在单链表中,取得第i个数据元素必须从头指针出发寻找,因此,单链表是非随机存取的存储结构。 建立单向循环链表的算法设计: LINK CreateL

9、inklist (DATA *d,int n) //单向循环链表的建立,d 输入函数返回值,n 节点个数; { NODE *l,*p,*s; // l 用来存放不带头结点的单链表的首节点的地址 ; int i; p=l=(NODE *)malloc(sizeof(NODE)); // 申请节点空间,p 指向第i+1个节点; p->num=d->num; //将第一个节点的信息转存在链表节点中; p->m=d->m; d++; //依次访问下一个数组元素; for(i=1;i

10、=(NODE *)malloc(sizeof(NODE)); //申请下一个节点空间,节点地址存放在s 中; s->num=d->num; s->m=d->m; p->next=s; p=s; d++; } s->next=l; //将首节点的地址存放在最后一个节点的next域形成环 return(l); // 返回单链表的首地址 } 链表建立流程图: 开始 int i; p=l=(NODE *)malloc(sizeof(NODE)); p->num=d->m;p->m=d->m;d++; inext=

11、l return(l) 结束 Y N i=1 s=(NODE *)malloc(sizeof(NODE)); s->num=d->m;s->m=d->m;d++; i++ 图2 建立单向循环链表流程图 2.3.5 Joseph环的构造 void Joseph ( LINk l, int n, int m) Joseph函数的作用是第一次从第一个节点开始,在循环链表中依次走过m个节点,将第m个节点的编号输出,并将第m 个节点中的m值付给m 。再从下一个节点出发,依次数m 个节点、、、、、如此循环直到所有节点编号都被输出时结束。该算法可理解为对特定结点进行删除,并将

12、所删结点的值进行输出。 在线性表链表中删除元素时,仅需改变所删除元素前个结点的指针,如p->next=p->nexe->next,可见,在单链表中删除结点时,只需修改指针而不需要移动元素。 删除链表中的结点e,并由系统收回其占用的存储空间,其过程如下: (1) 设定两个指针p和q。p指针指向被删除结点,q为跟踪指针,指向被删除结点的直接前驱结点。 (2) p从表头指针head指向的第一个结点开始依次向后搜索。当p->data等于e时,被删除结点找到。 (3) 修改p的前驱结点q的指针域。使p结点删除,然后释放存储空间。 在带头结点的单链线性表L中,删除第i个元素,并由e返回其值的算

13、法如下: Status ListDelete_L(LinkList &L,int i,ElemType &e) { p=L;j=0; While (p->next && jnext; ++j; } If ( ! ( p->next) || j>i-1) return ERROR; // 删除位置不合理 q=p->next; p->next=q->next; //删除并释放节点 e=q->data; free(q); return OK; } // List

14、Delete_L Joseph 环构造函数 void Joseph ( LINK l,int n,int m ) // l为单链表首节点地址,m是初始报数上限 {LINK p,q,s; int i,j; p=l; printf("\n"); for(i=0;inext; // p指向第m个节点 } printf("%d\t",p->num);

15、 //输出第m个节点的编号 m=p->m; q->next=p->next; //删除已经输出编号的节点 p=p->next; //从相邻的下一个节点开始下一次循环 } } 约瑟夫构造函数的流程图如下: 开始 int i,j; p=l i=0 inext 输出p->m m=p-m; q->next=p->next p=p->next; 结束 Y N Y N j++ 图3约瑟夫环构造函数流程图

16、 3源程序 #include #include typedef struct node { int num; int m; struct node *next; } NODE,*LINK; typedef struct data { int num; int m; } DATA; DATA * InputData(int *n) { DATA *d,*p,*s; int i; printf("How many people there are?\n"); printf("Please Input le

17、ngth:"); scanf("%d",n); d=(DATA *)malloc(*n*sizeof(DATA)); p=d; for(i=1;i<=*n;i++) { printf("Number%d ---- input m: ( m means the secret of them! ) :",i); scanf("%d",&p->m); p->num=i; p++; } return(d); } LINK CreateLinklist ( DATA *d, int n) { NODE *l,*p,*s ; int i; p

18、=l=( NODE * ) malloc ( sizeof (NODE) ); p->num=d->num; p->m=d->m; d++; for(i=1;inum=d->num; s->m=d->m; p->next=s; p=s; d++; } s->next=l; return(l); } void Joseph ( LINK l, int n, int m) { LINK p,q,s; int i,j; p=l; p

19、rintf("\n"); for(i=0;inext; } printf("%d\t",p->num); m=p->m; q->next=p->next; p=p->next; } } main() { int n,m; DATA *d; LINK l; d= InputData ( &n); l= CreateLinklist (d,n); printf("************************************\nPlease I

20、nput the initial value of m:\n************************************ "); scanf("%d",&m); Joseph(l,n,m); } 4运行结果及分析 4.1 运行结果 在main()主函数中,首先输出提示语句“How many people there are?\nPlease Input length: ”,这是输入人数n的值。接着调用输入数据函数InputDtata(),根据提示输入每个人的唯一密码m的值,编号默认为从1到n。输入完数据调用单链表建立函数Cr

21、eateLinklist()建立链表并将输入的人员信息存入链表里。这是界面显示语句“Please input the initial value of m:”,提示输入m的初始值。最后再调用约瑟夫环构造函数Joseph(),输出依次出列的人员编号,程序运行结束。截图如下。 图4 为每个人分配唯一密码 图5 输入m的初始值 图6 输出人员编号 为验证程序的正确性,再次输入一组很苛刻的数据,并运行分析。首先确定人数n的值为5,将所有人的唯一密码都设为0,初始m值也设为0。该组数据的理论输出序列为1,2,3,4,5。程序运行截图如下所示:

22、图7输入密码 图8 输出序列 4.2运行结果分析 首先输入人员个数为n=10,然后依次输入每个人的唯一编码,依次为2,9,0,3,7,5,11,23,46,17。m的初始值为65,从第一个人开始数最先输出的是编号为5的人,然后将该人的信息m值7付给m,从编号为6的人再开始数,输出的是编号为2的人,然后依此类推,输出编号为3,4,8,1,7,10,6,9的人。这与运行结果相吻合,也证实了结论的正确性。 第二组验证数据里,因为初始密码和每个人的密码均为0,所以指针p第一次指向的结点的编号即是要输出的编号,所以其输出序列和输入序列相同,为1,2,3,4,5。

23、 4.3程序评价 本程序先建立一个顺序表,用来存储数据。再构造单向循环链表,将数据存入链表节点里,之后依次为各个节点赋密码,并根据提示输入初始密码,然后,构造约瑟夫环函数,目的是对特定的节点进行删除,并将其值提取并输出。 本程序满足算法的5个重要特征,即有穷性、确定性、可行性、输入、输出。 程序不含语法错误,程序对于几组输入数据能够得出满足规定要求的结果,程序对于精心选择的典型、苛刻而带有刁难性的几组输入数据能够得出满足规格要求的结果,程序对于一切合法的输入数据都能产生满足规格要求的结果。 算法可读性较高,并在某些位置做了注释。 当输入数据非法时,算法也能适当地做出反应进行处

24、理,而不会产生莫名其妙的输出结果。 本程序的效率较高,时间复杂度为O(n)。 以上为本程序的一些优点,但是本程序还有一些不足之处,比如在单链表之外为玩家赋密码,还有引用data结构体等问题,这使得程序较为繁琐,希望下次设计时能够避免这些问题的发生。 5 设计体会 在这次课程设计中,我利用了链表一章的知识,解决了约瑟夫环问题,首先构造了单向循环链表,利用每个结点来模拟每一个玩家,又通过链表的赋值模拟为各人提供密码,借鉴链表的删除构造了约瑟夫环函数,并成功解决了问题。 这个设计的特点是程序源代码较少,但是思路复杂,根据题意我选则的是用循环链表解决问题,其实应用顺序表也可以解决这个

25、问题,不过后者较为复杂。 在完成任务的过程中我遇到了很过困难,比如如何构造约瑟夫环函数,但是我还是通过了手边的资料和同学的帮助克服了困难,我明白了成功的来之不易,以及成功之后的欣喜。 6 参考文献 [1] 李春葆. 数据结构教程[M]. 北京:清华大学出版社, 2006.10:P56-59 [2] 严蔚敏等.数据结构题集[M].北京:清华大学出版社, 2006.8:P89-90 [3] 朱国进.程序设计与问题求解[M].上海:东华大学出版社,2007,1 :P34-50 [4] 梁翎.语言程序设计实用技巧与程序实例[M].上海:上海科普出版社,2006.5 :P46

26、-50 [5] 陈国章. Turbo C程序设计技巧[M]. 天津:天津科学技术出版社, 2007.5:P78-90 [6] 王士元.C高级实用程序设计[M].北京:清华大学出版社,2007.6 :P29-43 [7] 陈明. 数据结构(C语言版)[M].北京:清华大学出版社,2006.3:P53-64 [8] 张群哲. 数据结构(C语言版)[M]. 西安: 西安电子科技大学出版社,2008.2:P28-43 [9] 顾为兵, 尹东, 袁平波, 朱明.数据结构及应用算法[M]. 合肥: 中国科学技术大学出版社, 2008.9:P33-36 [10] 李春葆.数据结构教程上机实验指导[M].北京:清华大学出版社,2006.7:P103-110

展开阅读全文
温馨提示:
1: 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
2: 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
3.本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
5. 装配图网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

相关资源

更多
正为您匹配相似的精品文档
关于我们 - 网站声明 - 网站地图 - 资源地图 - 友情链接 - 网站客服 - 联系我们

copyright@ 2023-2025  zhuangpeitu.com 装配图网版权所有   联系电话:18123376007

备案号:ICP2024067431-1 川公网安备51140202000466号


本站为文档C2C交易模式,即用户上传的文档直接被用户下载,本站只是中间服务平台,本站所有文档下载所得的收益归上传人(含作者)所有。装配图网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对上载内容本身不做任何修改或编辑。若文档所含内容侵犯了您的版权或隐私,请立即通知装配图网,我们立即给予删除!