约瑟夫问题(有时也称为约瑟夫斯置换,是一个出现在计算机科学和数学中的问题。在计算机编程的算法中,类似问题又称为约瑟夫环。又称“丢手绢问题”.)
问题来源:
据说著名犹太历史学家 Josephus有过以下的故事:在罗马人占领乔塔帕特后,39 个犹太人与Josephus及他的朋友躲到一个洞中,39个犹太人决定宁愿死也不要被敌人抓到,于是决定了一个自杀方式,41个人排成一个圆圈,由第1个人开始报数,每报数到第3人该人就必须自杀,然后再由下一个重新报数,直到所有人都自杀身亡为止。然而Josephus 和他的朋友并不想遵从。首先从一个人开始,越过k-2个人(因为第一个人已经被越过),并杀掉第k个人。接着,再越过k-1个人,并杀掉第k个人。这个过程沿着圆圈一直进行,直到最终只剩下一个人留下,这个人就可以继续活着。问题是,给定了和,一开始要站在什么地方才能避免被处决?Josephus要他的朋友先假装遵从,他将朋友与自己安排在第16个与第31个位置,于是逃过了这场死亡游戏。
算法分析:
1.数组法(也可利用指针实现)。利用数组存储n个编号,不断循环查找第m个要被杀的人,实现循环的关键就是对出现过的数据统一标记为0,利用某个变量记录间隔数目,找出满足要求的进行0标记,当循环从0到n时,使i=0,再次进入循环,直到寻找到的人数k==n,输出最后一个,即为生存者。代码如下:
#include<iostream>
using namespace std;
void main()
{
int n,m,a[101],k,i,j,num;
cout<<"请输入 人数(不超过100人) & 自杀的人之间的跨度:\n";
cin>>n>>m;
for(i=1;i<=n;i++)
{
a[i]=1;
}
j=0;
k=0;
for(i=1;i<=n;i++){
if(a[i]==1){
j=j+a[i];
if(j==m)
{
j=0;
a[i]=0;
cout<<i<<"->";
k++;
if(k%16==0)
cout<<'\n';
}
if(k==n){
num=i;
break;
}
}
if(i==n)
i=0;
}
cout<<"最后生存的是第"<<num<<"号!"<<endl;
cout<<"----------------------------------------"<<endl;
}
2.链表法
利用循环链表搭建整个联系网络,主要是链表的建立和删除,实现上就是将数组思想转化为链表思想。代码如下:
#include<iostream>
using namespace std;
struct Node{
int num;
struct Node *next;
}*Joseph;
int n,m;
void Create();
int del();
int main()
{
scanf("%d%d",&n,&m);
if(n==0)
{
cout<<"0人怎么玩。。。。。o(╯□╰)o\n";
return 0;
}
Create();
cout<<"\n\n生存者是第"<<del()<<"人\n\n";
return 0;
}
void Create()
{
int i=1;
Joseph=NULL;
Node *p,*q;
p=q=NULL;
while(i<=n)
{
p=new Node;
p->num=i++;
if(Joseph==NULL)
Joseph=p;
else
q->next=p;
q=p;
}
if(Joseph!=NULL)
q->next=Joseph;
}
int del()
{
int count=1;
Node *p,*q;
q=Joseph->next;
p=Joseph;
while(q!=q->next)
{
count++;
if(count%m==0)
{
p->next=q->next;
cout<<q->num<<"->";
delete q;
q=p->next;
}
else
{
p=q;
q=q->next;
}
}
cout<<q->num;
return q->num;
}