Josephus问题之Java乱弹篇
女儿生病住了几天医院,在我为女儿担心的同时也使我深深的感受到医疗产业化的伟大之处,在它一切向钱看的伟大精神指导之下,医生的工作积极性有了空前的提高,“开放思想,积极创收”成为各个医院的热门话题。我想大家也都发现了这种情况,那就是“没病当成有病看的多了,小病当成大病看的多了,大病当成重病看的多了”。为了让大家能看的起病,医疗产业化是不是到了该让它离开,还医院一f份圣洁的时候呢。
呵呵,说到离开,不禁让我想到编程中的一个算法问题:
说有10个小孩围成一圈做游戏,从第3个小孩起,顺时针方向数,每数到第5个小孩时,该小孩离开。小孩不断离开,圈子不断缩小,最后剩下的一个小孩便是胜利者。大家可能会说,这不就是Josephus问题吗。
不错,这就是Josephus问题。那大家知道这个问题的典故吗?
Josephus问题是建立在历史学家Joseph ben Matthias(成为Josephus)的一个报告的基础之上,该报告讲述了他和40个士兵在公元67年被罗马军队包围期间签定的一个自杀协定,Josephus建议每个人杀死他的邻居,他很聪明的使自己成为这些人中的最后一个,因此获得生还。呵呵,是不是觉得这个人很坏呢。
好了,说完这个典故,让我们来看看Josephus问题的具体算法实现吧。
public class Josephus {
public static void main(String args[])
{
int num = 10; //孩子总数
int interval = 5; //每次数interval个孩子,就让该孩子离开
int []child =new int[num+1]; //孩子树组
int []flag =new int[num+1]; //每个孩子是否在圈子的标志,1:在 0:不在
for(int i=1;i<=num;i++){
child[i]=i;
flag[i]=1; //开始每个孩子都在圈内
System.out.println("第"+i+"个孩子的名字:"+child[i]);
}
int n = 0;
int i = 3; //从第几个孩子开始
int j = 1; //从1开始记数
boolean noEnd = true; //是否结束的标志
while(noEnd)
{
while(j<interval){
i =(i+1>num?1:i+1);
j+=flag[i];
}
flag[i]=0;
n++;
if(n==num){
noEnd = false;
System.out.println("第"+i+"个孩子最后胜利");
}
else{
System.out.println("第"+i+"个孩子离开");
j=0; //j达到interval时,从新开始记数
}
}
}
}
大家可以改变数字试一下,挺有意思的。查看执行结果,我总觉得这里面似乎有什么规律,不过我还暂时没总结出来,如果那位朋友能总结出来,希望能让大家知道。
最后,把下面这几句我编的东东送给大家就算作个结束礼物吧。
医院不是电影院,不是想看就能看。
医院不是福利院,不是想拿就能拿,
医院不是养老院,不是想养就能养。
医院不是国务院,不是想进就能进。