约瑟夫问题算法及数据结构课程设计报告
实验报告——约瑟夫环

《数据结构》课程设计报告课程名称:《数据结构》课程设计课程设计题目:约瑟夫环姓名:张光栋院系:计算机学院专业:网络工程年级:2013级学号:13055532指导教师:张纪林一、需求分析1.以单项循环链表存储结构模拟约瑟夫环问题。
即编号为1、2、3…、n的n 个人按顺时针方向围坐一圈,每人持有一个密码(正整数)。
一开始任选一个正整数作为报数上限值m,从第一个人开始按顺时针方向自1开始报数,报到m时停止报数。
报m的人出列,将他的密码作为新的m值,从他在顺时针方向下一个人开始重新从1报数,如此下去,直至所有的人全部出列为止。
按出列顺序印出各人的编号。
2.演示程序以用户与计算机的对话方式执行,用户输入相应的数据,输出结果显示在其后。
3.测试数据:(1)n=55个人的密码依次为:2,4,2,6,2;首先m值为2(正确的输出顺序为:2 1 4 3 5)(2)n=77个人的密码依次为:2,4,1,4,3,2,3首先m值为5(正确的输出顺序为:5 1 3 4 6 2 7)二、概要设计为实现上述程序功能,可利用单向循环链表存储结构模拟此过程。
1.单向循环链表的抽象数据类型定义为:ADT CircularList{数据对象:D={ai|ai∈LNode,i=1,2,…,n,n≥0}数据关系:R1={<ai-1,ai>|ai-1∈D,i=2,…,n}基本操作:Status LisCreate_L(LinkList &L,int I,ElemType &e)操作结果:在不带头结点的单链表L中,创建第i个元素,并用e赋值}2.本程序中包括的两个基本模块:1)主程序模块:Void main(){初始化;do{接受命令;处理命令;}while(“命令”=”退出”)}2)循环链表模块:实现循环链表的抽象数据结构三、详细设计1.结点类型typedef struct ListNode{int mi;int n;struct ListNode *next;}ListNode,*LinkList;2.用循环链表存储约瑟夫环,没有头结点,基本操作函数如下:void CreateList(LinkList&L, int n){LinkList s;int i;L=(LinkList)malloc(sizeof(ListNode));L->n=1;L->next=L;for(i=2;i<=n;i++){s=(LinkList)malloc(sizeof(ListNode));s->next=L->next;L->next=s;s->n=i;L=L->next;}}void Delete(LinkList L, int m){int i;LinkList p,q;p=L;while(p->next!=p){for(i=1;i<m;i++)p=p->next;q=p->next;m=q->mi;printf("%d ",q->n);p->next=q->next;free(q);}printf("%d ",p->n);free(p);}3.主函数:int main(){int n,i,m;LinkList L,p;printf("请输入人数:");scanf("%d",&n);CreateList(L,n);printf("请输入密令\n");p=L->next;for(i=1;i<=n;i++){printf("请输入第%d条密令\n",i);scanf("%d",&p->mi);p=p->next;}printf("请输入初始密令\n");scanf("%d",&m);printf("输出为\n");Delete(L, m);return 0;}四、调试分析1.第一次写时,没有区分出只剩下的一个的情况,导致最终输出出现错误。
约瑟夫环问题实验报告

约瑟夫问题实验报告背景约瑟夫问题(Josephus Problem)据说著名犹太历史学家Josephus有过以下的故事:在罗马人占领乔塔帕特后,39 个犹太人与Josephus及他的朋友躲到一个洞中,39个犹太人决定宁愿死也不要被敌人到,于是决定了一个自杀方式,41个人排成一个圆圈,由第1个人开始报数,每报数到第3人该人就必须自杀,然后再由下一个重新报数,直到所有人都自杀身亡为止。
然而Josephus 和他的朋友并不想遵从,Josephus要他的朋友先假装遵从,他将朋友与自己安排在第16个与第31个位置,于是逃过了这场死亡游戏。
原题:用户输入M,N值,N个人围成一个环,从0号人开始数,数到M,那个人就退出游戏,直到最后一个人求最后一个剩下的人是几号?问题描述设编号为1-n的n(n>0)个人按顺时针方向围成一圈.首先第1个人从1开始顺时针报数.报m的人(m 为正整数).令其出列。
然后再从他的下一个人开始,重新从1顺时针报数,报m的人,再令其出列。
如此下去,直到圈中所有人出列为止。
求出列编号序列。
一.需求分析:(1)基本要求需要基于线性表的基本操作来实现约瑟夫问题需要利用循环链表来实现线性表(2)输入输出格式输入格式:n,m(n,m均为正整数,)输出格式1:在字符界面上输出这n个数的输出序列(3)测试用例(举例)输入:8,4输出:4 8 5 2 1 3 7 6二.概要设计(1)抽象数据类型:数据对象:n个整数数据关系:除第一个和最后一个n外,其余每个整数都有两个元素与该元素相邻。
基本操作:查找,初始化,删除,创建链表循环链表的存储结构:(2).算法的基本思想循环链表基本思想:先把n个整数存入循环链表中,设置第m个数出列,从第一个开始查找,找到第m个时,输出第m个数,并删掉第m个节点,再从下一个数开始查找,重复上一步骤,直到链表为空,结束。
(3).程序的流程程序由三个模块组成:1.输入模块:完成两个正整数的输入,存入变量n和m中2.处理模块:找到第m个数3.输出模块:按找到的顺序把n个数输出到屏幕上三.详细设计首先,设计实现约瑟夫环问题的存储结构。
实验报告 约瑟夫问题

pCur->next = pNew;
pCur = pNew;
printf("结点%d,密码%d\n",pCur->id, pCur->cipher);
}
}
printf("完成单向循环链表的创建!\n");
}
(3)运行"约瑟夫环"问题
static void StartJoseph(NodeType **, int)
exit(-1);
}
pNew->id = iId;
pNew->cipher = iCipher;
pNew->next = NULL;
return pNew;
}
(6)测试链表是否为空,空为TRUE,非空为FALSE
static unsigned EmptyList(const NodeType *pHead)
实验内容
利用循环链表实现约瑟夫环求解。
实验说明
1.问题描述
约瑟夫问题的:编号为1,2,....,N的N个人按顺时针方向围坐一圈,每人持有一个密码(正整数),一开始任选一个正整数作为报数上限值M,从第一个人开始按顺时针方向自1开始顺序报数,报到M时停止报数。报M的人出列,将他的密码作为新的M值,从他在顺时针方向上的下一个人开始重新从1报数,如此下去,直至所有人全部出列为止。试设计一个程序求出出列顺序。
{
if(!pHead)
{
return TRUE;
}
return FALSE;
}
实验中遇到的问题及解决方法
实验结果如下:
实验总结(结果和心得体会)
约瑟夫生死游戏(C++)数据结构实现

约瑟夫生死游戏(C++)数据结构实现本文档为约瑟夫生死游戏的C++数据结构实现文档,旨在详细介绍如何使用C++语言实现约瑟夫生死游戏的功能。
1:引言1.1 背景约瑟夫生死游戏是一个经典的数学问题,由约瑟夫·斯特恩提出。
问题描述如下:有n个人围成一圈,从某个人开始,每次顺时针报数m个人,报到m的人出局,直到剩下最后一个人为止。
1.2 目的本文档旨在指导开发人员使用C++语言实现约瑟夫生死游戏的功能,包括实现报数、出局等基本操作,并提供相应的测试样例和使用说明。
2:设计2.1 数据结构设计约瑟夫生死游戏的核心是一个环形链表,链表中的每个节点代表一个人。
每个节点包含两部分数据:该人的编号和指向下一个节点的指针。
链表的最后一个节点指向第一个节点,形成环形结构。
2.2 算法设计- 初始化链表:根据输入的人数创建相应数量的节点,并通过指针连接起来,形成环形链表。
- 报数出局:从指定的起始位置开始顺时针遍历链表,依次报数,当报到m时,将该节点从链表中移除。
- 判断游戏结束:当只剩下最后一个节点时,游戏结束。
2.3 功能设计- 初始化游戏:根据输入的人数和报数间隔,创建约瑟夫生死游戏实例。
- 开始游戏:执行报数出局操作,直到游戏结束。
- 获取胜利者:返回最后剩下的节点的编号。
3:实现下面给出C++语言实现约瑟夫生死游戏的核心代码。
```cppinclude<iostream>using namespace std;// 定义环形链表节点结构体struct Node {int id;Node next;};class JosephusGame {public:// 构造函数,初始化环形链表JosephusGame(int n, int m) {// 创建第一个节点Node firstNode = new Node;firstNode->id = 1;firstNode->next = NULL;// 依次连接剩余节点Node prevNode = firstNode;for (int i = 2; i <= n; i++) { Node newNode = new Node; newNode->id = i;newNode->next = NULL;prevNode->next = newNode;prevNode = newNode;}// 最后一个节点和第一个节点,形成环形结构prevNode->next = firstNode;// 初始化成员变量this->head = firstNode;this->count = n;this->interval = m;}// 游戏主循环void playGame() {Node currentNode = this->head;while (this->count > 1) {// 找到要出局的节点的前一个节点for (int i = 1; i < this->interval; i++) { currentNode = currentNode->next;}// 删除当前节点Node removedNode = currentNode->next; currentNode->next = removedNode->next; delete removedNode;this->count--;// 移动当前节点到下一个节点currentNode = currentNode->next;}}// 获取胜利者的编号int getWinner() {return this->head->id;}private:Node head; // 链表头节点int count; // 当前剩余人数int interval; // 报数间隔};int mn() {int n, m;cout << \。
数据结构实验一 约瑟夫环问题实验报告电子版

for(i = 1;i<length;i++){
tmp = (Node *)malloc(sizeof(Node));
tmp->number = num[i];
tmp->pass = pas[i];
pri->next = tmp;
pri = tmp;
pri->next = head;
for(i=0;i<time;i++){ //找到要删除的结点
tmp = tmp->next;
}
printf("%d ",tmp->number);
timeห้องสมุดไป่ตู้= tmp->pass - 1;
deleteFromList(&head,tmp);//删除结点
tmp = tmp->next;//从下一个结点又开始计算
initList(head);
createFromTail(head,num,pas,sizeof(num)/sizeof(num[0]));
p = head;
printf("\n约瑟夫计数前,每个数和他的密码:\n");
for(i = 0;i<sizeof(num)/sizeof(num[0]);i++){
}
}
// 从链表中删除
void deleteFromList(List *head,Node *tmp)
{
Node *tmp1;
Node *tmp2;
tmp1 = *head;
tmp2 = tmp1;
//如果链表剩了一个元素
约瑟夫环数据结构实验报告

约瑟夫环数据结构实验报告约瑟夫环数据结构实验报告引言约瑟夫环是一种经典的数学问题,它涉及到一个有趣的数据结构。
本次实验旨在通过实现约瑟夫环数据结构,深入理解该问题,并探索其在实际应用中的潜力。
本报告将介绍实验的设计和实现过程,并分析实验结果。
实验设计在本次实验中,我们选择使用链表来实现约瑟夫环数据结构。
链表是一种非常灵活的数据结构,适合用于解决约瑟夫环问题。
我们设计了一个Josephus类,其中包含了创建环、添加元素、删除元素等操作。
实验实现1. 创建环在Josephus类中,我们首先需要创建一个循环链表。
我们使用一个头节点来表示环的起始位置。
在创建环的过程中,我们可以选择指定环的长度和起始位置。
2. 添加元素在创建环之后,我们可以通过添加元素来向约瑟夫环中插入数据。
我们可以选择在环的任意位置插入元素,并且可以动态地调整环的长度。
3. 删除元素根据约瑟夫环的规则,每次删除一个元素后,下一个元素将成为新的起始位置。
我们可以通过删除元素的操作来模拟约瑟夫环的运行过程。
在删除元素时,我们需要考虑环的长度和当前位置。
实验结果通过实验,我们得出了以下结论:1. 约瑟夫环数据结构可以有效地模拟约瑟夫环问题。
通过创建环、添加元素和删除元素的操作,我们可以模拟出约瑟夫环的运行过程,并得到最后剩下的元素。
2. 约瑟夫环数据结构具有一定的应用潜力。
除了解决约瑟夫环问题,该数据结构还可以用于其他类似的问题,如任务调度、进程管理等。
3. 约瑟夫环数据结构的时间复杂度较低。
由于约瑟夫环的特殊性质,我们可以通过简单的链表操作来实现该数据结构,使得其时间复杂度较低。
结论本次实验通过实现约瑟夫环数据结构,深入理解了该问题,并探索了其在实际应用中的潜力。
通过创建环、添加元素和删除元素的操作,我们可以模拟出约瑟夫环的运行过程,并得到最后剩下的元素。
约瑟夫环数据结构具有一定的应用潜力,并且具有较低的时间复杂度。
通过本次实验,我们对数据结构的设计和实现有了更深入的理解,并为将来的研究和应用奠定了基础。
数据结构课程设计约瑟夫环

《数据结构》课程设计报告书题目:约瑟夫环系别:计算机科学与应用学号:学生姓名:指导教师:完成日期:2012年6月7日目录1.需求分析 (3)1.1 功能分析 (3)1.2开发平台 (3)2.概要设计 (3)3. 程序设计主要流程 (5)4.调试与操作说明 (5)4.1调试情况 (5)4.2操作说明 (6)总结 (8)致谢 (9)附录 (9)参考文献 (13)指导教师评语: (14)1.需求分析1.1 功能分析本次选做的课程设计是改进约瑟夫(Joseph)环问题。
约瑟夫环问题是一个古老的数学问题,本次课题要求用程序语言的方式解决数学问题。
此问题仅使用单循环链表就可以解决此问题。
在建立单向循环链表时,因为约瑟夫环的大小由输入决定。
为方便操作,我们将每个结点的数据域的值定为生成结点时的顺序号和每个人持有的密码。
进行操作时,用一个指针r指向当前的结点,指针H指向头结点。
然后建立单向循环链表,因为每个人的密码是通过scanf()函数输入随机生成的,所以指定第一个人的顺序号,找到结点,不断地从链表中删除链结点,直到链表剩下最后一个结点,通过一系列的循环就可以解决改进约瑟夫环问题。
1.2开发平台WindowsXP操作系统;Microsoft Visual C++ 6.0;2.概要设计编号为1,2… n的n个人按顺时针方向围坐一圈,每人持有一个密码(正整数)。
一开始任选一个正整数作为报数的上限值m,从第一个人开始按顺时针方向自1开始顺序报数,报到m时停止报数,报m的人出列,将他的密码作为新的m值,从他的顺时针方向上的下一个开始重新从1报数,如此下去,直至所有人全部出列为止,设计一个程序求出出列顺序。
这个问题采用的是典型的循环链表的数据结构,就是将一个链表的尾元素指针指向队首元素。
r->next=H。
解决问题的核心步骤:首先建立一个具有n个链结点,无头结点的循环链表。
然后确定第1个报数人的位置。
最后不断地从链表中删除链结点,直到链表为空。
数据结构joseph课程设计

数据结构joseph课程设计一、课程目标知识目标:1. 学生能理解约瑟夫问题(Josephus problem)的背景和数学原理,掌握其与数据结构中循环链表的关系。
2. 学生能够掌握循环链表的基本操作,包括节点的插入、删除以及遍历。
3. 学生能够运用所学的循环链表知识解决约瑟夫问题,并理解其算法的效率。
技能目标:1. 学生能够运用编程语言(如C/C++/Java等)实现循环链表,并完成相应的约瑟夫问题求解程序。
2. 学生通过实际操作循环链表,提高逻辑思维能力和编程实践能力。
3. 学生能够通过分析、讨论和解决问题,培养团队协作能力和问题解决能力。
情感态度价值观目标:1. 学生通过解决实际问题,增强对数据结构学习的兴趣和热情,形成积极向上的学习态度。
2. 学生在团队协作中学会尊重他人,培养良好的沟通能力和合作精神。
3. 学生通过探究和解决约瑟夫问题,体会数学和计算机科学的实际应用价值,增强对科学的敬畏之心。
课程性质:本课程设计属于数据结构学科范畴,以实践操作和问题解决为核心,强调理论与实践相结合。
学生特点:考虑到学生已具备一定的编程基础和逻辑思维能力,课程设计将注重培养学生的实践能力、团队协作能力以及创新意识。
教学要求:教师应关注学生的个体差异,因材施教,引导学生通过自主探究、合作学习等方式达到课程目标。
在教学过程中,注重过程评价和结果评价相结合,全面评估学生的学习成果。
二、教学内容本节教学内容围绕数据结构中的循环链表及其应用——约瑟夫问题展开,具体安排如下:1. 循环链表基础知识回顾:- 循环链表的定义与特点- 循环链表的节点结构- 循环链表与普通链表的区别2. 循环链表的操作:- 节点的插入与删除- 循环链表的遍历- 循环链表的应用场景3. 约瑟夫问题介绍:- 约瑟夫问题的背景和数学原理- 约瑟夫问题与循环链表的关系4. 约瑟夫问题求解:- 算法设计思路- 编程实现步骤- 算法效率分析5. 实践环节:- 编写循环链表的基本操作函数- 编写求解约瑟夫问题的程序- 调试与优化程序6. 教学案例分析:- 结合实际案例,讲解循环链表在解决约瑟夫问题中的应用- 分析案例中的算法优化方法教学内容根据课本相应章节进行组织,确保学生能够在掌握循环链表基础知识的基础上,学会解决实际问题。
- 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
- 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
- 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。
线性表的操作及其应用一、问题描述线性表、队列是一种常用的数据结构,有顺序和链式两种存储结构,在实际中应用十分广泛,而链表又分为单链表和循环链表,队列又分为链式队列和循环队列。
这些数据结构都可用来解决约瑟夫环问题。
约瑟夫环问题是算法设计中的一个经典问题,是顺序编号的一组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、void InitQueue(SqQueue *Q)初始化循环队列2、void DestroyQueue(SqQueue *Q)销毁循环队列3、void ClearQueue(SqQueue *Q)清空循环队列4、int QueueEmpty(SqQueue Q)判断空队列5、int QueueLength(SqQueue Q)求循环队列长度6、void GetHead(SqQueue Q, ElemType *e)取队头元素7、int EnQueue(SqQueue *Q, ElemType e)入队列8、int DeQueue(SqQueue *Q, ElemType *e)出队列9、void QueueTraverse(SqQueue Q)遍历循环队列并输出元素10、void InitQueue(LQueue *Q)初始化队列11、void DestroyQueue(LQueue *Q)销毁队列12、void ClearQueue(LQueue *Q)清空队列13、int QueueEmpty(LQueue Q)判断空队列14、int QueueLength(LQueue Q)求队列长度15、ElemType GetHead(LQueue Q, ElemType *e)取队头元素16、void EnQueue(LQueue *Q, ElemType e)入队列17、void DeQueue(LQueue *Q, ElemType *e)出队列18、void QueueTraverse(LQueue Q)遍历队列并输出元素19、void joseffer(int n)约瑟夫环20、int CreateList(LinkList &L,int n)建立顺序链表21、void josephus_clist(LinkList &L,int m)顺序链表解决约瑟夫环22、void InitList(SqList *L)初始化链表23、void ysf1()链表解决约瑟夫环24、void ysf2()循环链表解决约瑟夫环25、void ysf3(){链式队列解决约瑟夫环问题26、void ysf4()循环队列解决约瑟夫环问题27、void menu()菜单28、int main()主函数六、数据结构//(ADT)typedef int ElemType;/* 定义顺序表类型*/typedef struct{ ElemType *elem;int length;} SqList;/* 定义循环链表类型*/typedef struct LNode{ int num;int code;struct LNode *next;} *LinkList;/* 定义循环队列类型*/typedef struct{ ElemType *base;int front;int rear;} SqQueue;/* 定义链式队列类型*/typedef struct QNode{ ElemType data;struct QNode *next;} QNode,*QueuePtr;typedef struct{ QueuePtr front;QueuePtr rear;} LQueue;七、源程序#include "stdlib.h"#include "stdio.h"#define N 100typedef int ElemType;/* 定义顺序表类型*/typedef struct{ ElemType *elem;int length;} SqList;/* 定义循环链表类型*/typedef struct LNode{ int num;int code;struct LNode *next;} *LinkList;/* 定义循环队列类型*/typedef struct{ ElemType *base;int front;int rear;} SqQueue;/* 定义链式队列类型*/typedef struct QNode{ ElemType data;Struct QNode *next;} QNode,*QueuePtr;typedef struct{ QueuePtr front;QueuePtr rear;} LQueue;/* 初始化循环队列*/void InitQueue(SqQueue *Q){ Q->base=(ElemType*)malloc(N*sizeof(ElemType));Q->front=Q->rear=0; }/* 销毁循环队列*/void DestroyQueue(SqQueue *Q){ free(Q->base); }/* 清空循环队列*/void ClearQueue(SqQueue *Q){ Q->front=Q->rear=0; }/* 判断空队列*/int QueueEmpty(SqQueue Q){ if (Q.front==Q.rear)return 1;elsereturn 0; }/* 求循环队列长度*/int QueueLength(SqQueue Q){ return (Q.rear+N-Q.front)%N; }/* 取队头元素*/void GetHead(SqQueue Q, ElemType *e){ if (Q.front!=Q.rear)*e=Q.base[Q.front];}/* 入队列*/int EnQueue(SqQueue *Q, ElemType e){ if ((Q->rear+1)%N==Q->front)return 0;Q->base[Q->rear]=e;Q->rear=(Q->rear+1)%N;return 1; }/* 出队列*/int DeQueue(SqQueue *Q, ElemType *e){ if (Q->front==Q->rear)return 0;*e=Q->base[Q->front];Q->front=(Q->front+1)%N;return 1; }/* 遍历循环队列并输出元素*/void QueueTraverse(SqQueue Q){ int i;printf("\nQueue: ");if (Q.rear<Q.front) Q.rear=Q.rear+N;for(i=Q.front; i<Q.rear; i++)printf("%d\t",Q.base[i%N]);}/* 初始化队列*/void InitQueue(LQueue *Q){ Q->front=Q->rear=(QueuePtr)malloc(N*sizeof(QNode));if(!(Q->front)) exit(0);Q->front->next=NULL;}/* 销毁队列*/void DestroyQueue(LQueue *Q){ while(Q->front){ Q->rear=Q->front->next;free(Q->front);Q->front=Q->rear;}}/* 清空队列*/void ClearQueue(LQueue *Q){ QueuePtr p;p=Q->front->next;while(p){ Q->front->next=p->next;free(p);Q->rear=Q->front;}}/* 判断空队列*/int QueueEmpty(LQueue Q){ if (Q.front==Q.rear)return 1;elsereturn 0; }/* 求队列长度*/int QueueLength(LQueue Q){ QueuePtr p; int n=0;p=Q.front;while(p!=Q.rear){ n++;p=p->next;}return n;}/* 取队头元素*/ElemType GetHead(LQueue Q, ElemType *e){ if (Q.front!=Q.rear)return Q.front->next->data;else return 0;}/* 入队列*/void EnQueue(LQueue *Q, ElemType e) { QueuePtr p;p=(QueuePtr)malloc(N*sizeof(QNode));if(!p) exit(0);p->data=e;p->next=NULL;Q->rear->next=p;Q->rear=p;}/* 出队列*/void DeQueue(LQueue *Q, ElemType *e) { QueuePtr p;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);}}/* 遍历队列并输出元素*/void QueueTraverse(LQueue Q){ QueuePtr p;printf("\nQueue:");p=Q.front->next;while(p){ printf("%d\t",p->data);p=p->next;}}/*约瑟夫环*/void joseffer(int n){ LQueue Q;int i;ElemType x;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);else printf("%4d",x);}}}/*建立顺序链表*/int CreateList(LinkList &L,int n){ LNode *p,*q;printf("Input %d number:\n",n);L=q=(LinkList)malloc(sizeof(LNode));if(L==NULL || q==NULL)return 0;for(int i=1;i<=n;i++){ p=(LinkList)malloc(sizeof(LNode));scanf("%d",&p->code);p->num=i;if(i==1)L=p;elseq->next=p;q=p;}p->next=L;L=p;return 1;}/*顺序链表解决约瑟夫环*/void josephus_clist(LinkList &L,int m){ int e=m,k;LinkList p,pre,tp;p=L;while(p!=NULL){ for(int j=0;j<e;j++){ pre=p;p=p->next;}e=p->code;k=p->num;printf("The output number is %d\n",e);if(pre!=p){ tp=p;pre->next=p->next;p=p->next;p=pre;free(tp);}else{ free(pre);p=NULL;}}}/*初始化链表*/void InitList(SqList *L){ L->elem=(ElemType*)malloc(N*sizeof(ElemType));L->length=0; }/*链表解决约瑟夫环*/void ysf1(){ SqList L;int m,i,d,k;InitList(&L);printf("元素个数和报到第几出");scanf("%d%d",&L.length,&d);printf("输入元素");for(i=0;i<L.length;i++)scanf("%d",&L.elem[i]);m=0;k=0;i=0;while(m<L.length){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;}}/*循环链表解决约瑟夫环*/void ysf2(){ int m,n;LinkList jos_clist;printf("Input the beginging number:\n");scanf("%d",&m);printf("Input n people:\n");scanf("%d",&n);CreateList(jos_clist,n);josephus_clist(jos_clist,m);}/*链式队列解决约瑟夫环问题*/void ysf3(){ LQueue Q;int i; ElemType x; int n;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);}/*循环队列解决约瑟夫环问题*/void ysf4(){ SqQueue Q; int i,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;}}/*菜单*/void menu(){ int choice;do{printf("\n====================================");printf("\n 主菜单");printf("\n 1.顺序表实现约瑟夫环问题");printf("\n 2.循环链表表实现约瑟夫环问题");printf("\n 3.链式队列实现约瑟夫环问题");printf("\n 4.循环队列实现约瑟夫环问题");printf("\n 5.退出程序");printf("\n====================================");printf("\n 请输入您的选择(1,2,3,4,5) choice=");scanf("%d",&choice);switch(choice){ case 1:ysf1();break;case 2:ysf2();break;case 3:ysf3();break;case 4:ysf4();break;case 5:exit(0);}} while(choice<=5);}/*主函数*/int main(){ menu();System("pause");return 0;}八、测试情况程序的测试结果如下:1、顺序表实现约瑟夫环问题测试结果正确。