(8) 约瑟夫问题总结

Josephus问题:假设有n个人排成一个圈,从第一个人开始报数,数到第m个人的时候这个人从圈里出列,然后继续在环里数到第m个人,让其出列,直到所有人都出列,求最后圈里剩下的那个人,在以前n个人里是第几个?


其中,有许多类似的问题,都属于约瑟夫问题,如:

         500个小孩围成一圈,从第一个开始报数:1,2,3,1,2,3,1,2,3……每次报3的小孩退出,然后再从1开始报数,问最后剩下的那个小孩,在以前500人里是第几个?


package test;

import java.util.ArrayList;
import java.util.Iterator;
import java.util.LinkedList;
import java.util.List;
import java.util.Queue;

/**
 * 约瑟夫问题 500个小孩围成一圈,从第一个开始报数:1,2,3,1,2,3,1,2,3,……每次报3的小孩退出,然后再从1开始报数,
 * 问最后剩下的那个小孩,在以前500人里是第几个?
 *
 * 思路1:数组,可以通过ArrayList实现,将所有的小孩放到ArrayList里,然后判断小孩的位置是否是3,是的话,将其remove掉;
 * 思路2:链表,可以通过LinkedList实现; 
 * 思路3:队列,可以通过LinkedList实现;
 * 思路4:递归;
 * 思路5:循环;
 */
public class Josephus {

	/**
	 * 数组实现第一种方式
	 * @param num
	 * @param quitNum
	 * @return
	 */
	public static int baoshu2(int num, int quitNum) {
		List<Integer> lists = new ArrayList<Integer>();

		for (int i = 1; i <= num; i++) {
			lists.add(i);
		}
		int index = 0;
		while (lists.size() > 1) {
			for (int i = 0; i < lists.size(); i++) {
				index++;
				if (index % quitNum == 0) {
					index = 0;
					lists.remove(i);
					i--;
				}
			}
		}
		return lists.get(0);
	}

	/**
	 * 数组实现第二种方式
	 * @param num
	 * @param quitNum
	 * @return
	 */
	public static int baoshu(int num, int quitNum) {

		// 用来放置n个小孩,list每个元素的值,存放小孩的初始位置
		List<Integer> list = new ArrayList<Integer>();
		// 初始化list
		for (int i = 1; i <= num; i++) {
			list.add(i);
		}
		// 小孩开始报数,数3的小孩出来,index代表小孩的当前位置
		int index = 0;
		while (list.size() > 1) {
			// 当前位置加上报数就是报数后的位置
			index = (index + quitNum - 1) % list.size();
			// 移除当前小孩
//			System.out.println(index + "--" + list.get(index));
			list.remove(index);
		}
		return list.get(0);
	}

	/**
	 * 链表实现
	 * @param num
	 * @param quitNum
	 * @return
	 */
	public static int getCountOffResult(int num, int quitNum) {
		// 入口参数检查
		if (num <= 0 || quitNum <= 0) {
			return 0;
		}

		// 定义并生成小孩数组列表,编号从1到numberOfChildren
		List<Integer> children = new LinkedList<>();
		for (int ii = 1; ii <= num; ii++) {
			children.add(ii);
		}

		Iterator<Integer> childrenIterator = children.iterator();
		while (num > 1) {
			Integer childNo = 0;
			for (int ii = 0; ii < quitNum; ii++) {
				// 判断链表是否到了最后一个节点
				if (!childrenIterator.hasNext()) {
					childrenIterator = children.iterator();
				}
				childNo = childrenIterator.next();
			}

			childrenIterator.remove();
			num--;
		}

		childrenIterator = children.iterator();
		return childrenIterator.next();
	}

	/**
	 * 队列,使用LinkedList实现,因为LinkedList实现了Queue接口
	 * 
	 * @param num 小孩总数
	 * @param quitNum 第几个小孩
	 * @return
	 */
	public static int queue(int num, int quitNum) {
		Queue<Integer> q = new LinkedList<Integer>();
		for (int i = 1; i <= num; i++) {
			q.add(i);
		}
		// 将数1、2的小伙伴放到队尾,将数3的小伙伴扔出去
		while (q.size() > 1) {
			for (int i=1;i < quitNum ;i++) {
				q.add(q.remove());
			}
			q.remove();
		}
		// 打印出最后一个小伙伴的位置
		Integer position = 0;
		if ((position = q.poll()) != null) {
			return position;
		} else {
			return 0;
		}
	}

	/**
	 * 递归实现
	 * 
	 * @param num  小孩总数
	 * @param quitNum 第几个小孩
	 * @param i 次数
	 * @return
	 */
	public static int fun(int num, int quitNum, int i) {
		if (i == 1) {
			return (num + quitNum - 1) % num;
		} else {
			return (fun(num - 1, quitNum, i - 1) + quitNum) % num;
		}
	}

	/**
	 * 约瑟夫循环非递归解法 
	 * 思路:采用倒推方式
	 * 
	 * @param num 小孩总数
	 * @param quitNum 第几个小孩
	 * @return
	 */
	public static int getLastChild(int num, int quitNum) {
		int index = 0;
		for (int i = 2; i <= num; i++) {
			index = (index + quitNum) % i;
		}

		return index + 1;
	}

	public static void main(String[] args) {

		System.out.println(fun(400, 3, 400) + 1);
		System.out.println(baoshu2(400, 3));
		System.out.println(baoshu(400, 3));
		System.out.println(queue(400, 3));
		System.out.println(getCountOffResult(400, 3));
		System.out.println(getLastChild(400, 3));

	}
}

其中程序参考自:

约瑟夫问题讨论

Josephus环问题


原文地址:https://www.cnblogs.com/xiaozhang2014/p/5297282.html