约瑟夫环(已知N个人,围坐在一张圆桌周围)

已知N个人(以编号1,2,3,4,...,N分别表示)围坐在一张圆桌周围。从编号为K的人开始报数,数到M的那个人出列,他的下一个人又从K开始报数,数到M的那个人又出列,依此规律重复下去,直到圆桌周围的人全部出列,试用C++编程实现。

分析问题是一个约瑟夫环的问题,使用循环链表。

//创建循环链表

typedef struct LNode{
int data;
struct LNode *link;}LNode,*LinkList;
void yuesefu(int n,int k,int m)
{
LinkList p,list,curr;
p=(LinkList)malloc(sizeof(LNode));
p->data=0;
p->link=p;
curr=p;
for(int i=1;i<n;i++)
{
LinkList t;
t=(LinkList)malloc(sizeof(LNode));
t->data=i;
t->link=curr->link;
curr->link=t;//循环链表
curr=t;
}
//算法

while(k--)list=p,p=p->link;
while(n--)
{
for(int s=m-1;s--;list=p,p=p->link);
list->link=p->link;
cout<<p->data<<"->";
free(p);
p=list->link;
}





  • 0
    点赞
  • 16
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 这是一道约瑟夫环问题:已知n个人(以编号1,2,3...n分别表示)围坐在一张圆桌周围。从编号为1的人开始报数,数到m的那个人出列;他的下一个人又从1开始报数,数到m的那个人又出列;依此规律重复下去,直到圆桌周围的人全部出列。求出列顺序(以编号形式表示)。 可以通过模拟这个过程的方式解决,首先我们创建一个长度为n的数组,表示圆桌上的人,初始化每个人的状态为存活状态(0表示存活,1表示死亡)。 然后从编号为1的人开始报数,这个人报数完毕后需要判断他是否应该出列。如果他应该出列,那么将他的状态修改为死亡状态,并将他的编号加入到结果数组中。否则,将他的编号加入到一个队列中,表示他还没有数到m的位置。接着,从队列中取出下一个编号,继续进行报数,重复这个过程直到所有人都出列为止。 最后返回结果数组即可。 ### 回答2: 约瑟夫环是一个经典的数学问题,它可以让我们看到人在面对困难时的选择和决策能力。 问题是这样的:有n个人围成一个圆圈,从第一个人开始报数,报到第m个人出圈,并从下一个人开始报数,重复这个过程直到只剩下一个人,这个人就是胜利者。这个问题的解法是在1~n之间循环求解,每次删去第m个人,直到只剩下一个人。 对于这个问题,我们可以使用模拟法进行解决,即模拟这个过程,直到只剩下一个人为止。我们可以用数组将这n个人的编号储存起来,然后按照题意循环删除,每次循环判断是否到达数组末尾,如果到达末尾则回到数组头部继续删除,直到只剩下一个人为止。 除此之外,我们还可以使用递推公式进行解法。设f(n,m)表示n个人报数,每报到m就删除一个人最后剩下的人的编号。由于每一次都删除了一个人,所以到了n-1个人时,我们可以知道到底是哪一个人被删除掉了,即设编号为x。则我们可以将n个人的编号重新编号为1, 2, 3, ..., n-1,并且将x号的人作为编号为1的人重新开始报数,这样就可以得到n-1个人的解f(n-1, m)。然后我们可以将n个人的编号重新编号为2, 3, 4, ..., n,并将x号的人作为编号为2的人重新开始报数,得到一个新的问题的解f(n-1, m)。然后我们可以用递归来解决整个问题,即f(n, m) = (f(n-1, m) + m) % n。当n=1时,f(n, m)的值即为最后剩下的人的编号。 ### 回答3: m的那个人又出列;依此规律重复下去,直到圆桌周围的人全部出列。而zzulioj是一个在线测评平台,提供了关于约瑟夫环的一道编程题目。 对于这道题目,我们需要编写一个模拟算法来解决。首先,我们可以将所有的人按顺序排列起来,并通过数组保存每个人的编号。然后,我们从编号为1的人开始,按照题目规则不断地找到应该出列的人,并将其从列表中删除。直到所有人都被删除为止。 具体算法如下: 1. 初始化列表,将所有人的编号保存到一个数组中。 2. 从编号为1的人开始,按照题目规则不断地找到应该出列的人:将当前位置加上m,并对列表长度取模,即可找到应该出列的人。 3. 将出列的人从列表中删除。 4. 当列表中只剩一个人时,结束循环。 5. 输出最终剩下的那个人的编号。 通过这个算法,我们可以解决约瑟夫环问题,并且在zzulioj上进行测试。同时,这个算法也可以用来解决其他类似的问题,如“圆桌上20个人报数,报到13的人出圈……”等问题。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值