ZhenyuGo

导航

有1到100共100个数, 从1开始, 每隔1, 2, 3... 个数拿走一个数, 最后剩下几?(约瑟夫环)

最近找实习, 在做Test Assignment时遇到了这么道题, 就顺便记录下来:
说, 有1到100共100个数, 摆成一个圈. 从1开始, 每隔1, 2, 3, 4 ... 个数拿走一个数, 一直循环, 最后剩下几? 具体的讲就是一开始(隔0个数)把 1 拿走, 隔1个数(2)把3拿走, 再隔2个数(4, 5)把6拿走, 再隔3个数(7, 8, 9)把10拿走. 第一圈数到100之后接着从2开始数, 直到最后剩下1个数为止, 请问最后剩下几? 如果是1到n呢?

 1     public static int selectNumber(int n) {
 2         
 3         int result = -1;
 4         ArrayList<Integer> nums = new ArrayList<Integer>();
 5         int index = 0;     // this variable is used to remove numbers from list 
 6         int count = 0;     // count is used to count which numbers should be remove
 7         int pIndex = 0;  // this is used to record previous index
 8         
 9         // generate a list contains numbers from 1 to n
10         for(int i = 1; i <= n; i++) {
11             nums.add(i);
12         }
13         
14         while(nums.size() > 1) {
15             
16             while(index < nums.size()) {
17                 nums.remove(index);     
18                 count++;
19                 pIndex = index;
20                 index += count;
21             }
22             
23             index = count - (nums.size() - pIndex);
24             
25             while(index > nums.size() - 1) {
26                 index = index - nums.size();
27             }
28         }
29         
30         result = nums.get(0);
31         return result;
32     }
33     
34     public static void main(String[] args) {
35         int surviver = selectNumber(100);
36         System.out.println("The surviver is: " + surviver);
37     }

以上就是我的解决方案, 最后留下的数是31


2016.06.04更新:

在我发布这篇博文之后热心网友指出这叫"约瑟夫环". 于是我就去网上搜了一下, 并找到了一个比较简洁的约瑟夫环代码(这个代码不是我原创, 是在别人代码的基础上修改得来):

 1     /**
 2      * 约瑟夫环:
 3      * n个数字摆成一个环, 从1开始数, 数到m或m的倍数的数字被删除, 然后继续, 直到剩下1个数字为止
 4      */
 5     public static int josephProbBasic(int n, int m) {
 6         ArrayList<Integer> list = new ArrayList<>();
 7         for(int i = 1; i <= n; i++) {
 8             list.add(i);
 9         }
10         // 用于计数
11         int count = 1;
12         // 当list中只剩下1个数时, 结束循环
13         for(int i = 0; list.size() > 1; i++) {
14             // 如果数到list的结尾, 从0开始重新数
15             if(i == list.size()) {
16                 i = 0;
17             }
18             // 在m的倍数时, 删除对应的数
19             if(count % m == 0) {
20                 // 由于删除之后, list中被删除元素之后的元素都会依次向前移动一位, 因此也要把i向前移动一位
21                 list.remove(i--);
22             }
23             count++;
24         }
25         return list.get(0);
26     }

但是这只是普通的约瑟夫环, 而我的问题要更麻烦一点. 于是乎, 我根据上面的思路重写了这个问题的代码:

 1     /**
 2      * 升级版约瑟夫环:
 3      * n个数字摆成一个环, 从1开始数, 每隔m个数字删除一个数字. m的初始值为0, 然后m++, 就是0, 1, 2, 3....
 4      *  这样不断删除, 直到剩下1个数字为止.
 5      * @param n
 6      * @return
 7      */
 8     public static int josephProbV2(int n) {
 9         ArrayList<Integer> list = new ArrayList<>();
10         for(int i = 1; i <= n; i++) {
11             list.add(i);
12         }
13         int count = 0;
14         int m = 0;
15         for(int i = 0; list.size() > 1; i++) {
16             if(i == list.size()) {
17                 i = 0;
18             }
19             // 如果count数到m个数, 就删除对应的数字, 同时count从零开始重新数, m = m + 1
20             if(count == m) {
21                 list.remove(i--);
22                 count = 0;
23                 m++;
24             }else {
25                 count++;
26             }
27         }
28         return list.get(0);
29     }

这样感觉清爽多了. 

 

posted on 2016-05-26 08:30  ZhenyuGo  阅读(2653)  评论(3编辑  收藏  举报