约瑟夫问题(经典问题)

约瑟夫问题(有时也称为约瑟夫斯置换,是一个出现在计算机科学数学中的问题。在计算机编程的算法中,类似问题又称为约瑟夫环。又称“丢手绢问题”.)

问题来源:

据说著名犹太历史学家 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;
}

你可能感兴趣的:(循环链表,约瑟夫问题,猴子选大王,经典题目)