经典问题-约瑟夫环

经典问题-约瑟夫环

问题描述

约瑟夫环就是n个人围在一起丢手绢。手绢每经过一个人报一个数,开始的时候从第一个人开始报数1,报数依次增加。若报到3的人出局,并且下一个人重新从1开始报数。直到局内剩下两个人。

问题分析

这个问题的输入有两个。第一个是n,参与游戏的人数。第二个是m,最后留下的幸存者人数。
输出有一行,是幸存者的编号。
这个问题有两种解题思路。第一种是递归法,找到递推关系。第二种则是依靠循环链表进行模拟。
实际上,这是同一个过程从前后两个方向进行推导。

解决方案

基于循环链表的解决方案便是建立一个有n个节点的循环链表,模拟其过程,得到答案。
对此,我们只需要一个指针指向第一个节点。在每次报到m时,删除当前指针指向的节点,并且下一个从1
开始报号。

public static LinkedList<Integer> josephus(int n, int m) {
	LinkedList<Integer> jos = new LinkedList<Integer>();
	for (int i = 1; i <= n; ++i) {
		jos.add(i);
	}
	Iterator<Integer> it = jos.iterator();
	int i = 0, a = m;
	while (jos.size() >= m) {
		if (!it.hasNext()) {
			it = jos.iterator();
		}
		i++;
		if (i == a) {
			i = 0;
			jos.removeByElement(it.next());
		} else {
			it.next();
		}
	}
	return jos;
}

递归的核心便是找到每次报数时,幸存者的序号变化规律。

for (int i = m - 1; i <= n; ++i){
	res = (res + m) % i;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值