<script type="text/javascript">
</script> <script type="text/javascript">
</script>
方法一
- /* 用常规方法实现(双向回环链表算法)
- *
- * 逢三退一:一个头和尾相连的圆形数字队列,从第一个开始数,数到三,
- * 就把三的那个去掉,然后接着数,如此循环,逢三退一。
- * 直到最后一个为止,求最后一个是数字队列中的第几个?
- */
- public class Cout3Quit
- {
- public static void main(String[] args)
- {
- boolean [] arr = new boolean [ 500 ]; // 创建一个布尔型数组
- int contNum = 0 ; // 数数,计算数了多少个,数到3就退1个
- int index = 0 ; // 第几个位置,数到的位置
- int arrlength = arr.length; // 数字队列还有多少个
- for ( int i = 0 ; i < arr.length; i++)
- {
- arr[i] = true ; // 把数组的值得全部置为true
- }
- while (arrlength > 1 ) // 留下最后一个
- {
- if (arr[index] == true )
- {
- contNum++; // 从0开始数
- if (contNum == 2 ) // 从0开始的,数到2就是第三个数字了
- {
- arr[index] = false ; // 数到第三个数,把它的值变为false
- arrlength--; // 数组长度减去1
- contNum = 0 ; // 重新数数
- }
- }
- index++; // 每判断一个,加1
- if (index == arr.length) // 判断是否到了最后的数字,到了,重新从0开始
- {
- index = 0 ;
- }
- }
- // 打印数组中有true值的那个下标值,即最后那个是第几个
- for ( int j = 0 ; j < arr.length; j++)
- {
- if (arr[j] == true )
- {
- System.out.println(j);
- }
- }
- }
- }
方法二
- /*
- * 用面向对象方法实现(双向回环链表算法)
- *
- * 逢三退一:一个头和尾相连的圆形数字队列,从第一个开始数,数到三,
- * 就把三的那个去掉,然后接着数,如此循环,逢三退一。
- * 直到最后一个为止,求最后一个是数字队列中的第几个?
- */
- public class Count3Quit2
- {
- public static void main(String[] args)
- {
- KidCircle kc = new KidCircle( 500 );
- int num = 0 ;
- Kid k = kc.first; //从第一个开始数数
- while (kc.count > 1 )
- {
- num++; //数数
- if (num == 2 ) // 从0开始的,数到2就是第三个数字了
- {
- kc.delete(k); //删除当前的k
- num = 0 ; //重新数数
- }
- k = k.right; //k右孩子赋给k
- }
- System.out.println(kc.first.id); // 打印最后的一个孩子的id
- }
- }
- class Kid
- {
- int id = 0 ; // 孩子号码
- Kid left; // 左边的孩子
- Kid right; // 右边的孩子
- }
- class KidCircle
- {
- int count = 0 ;
- Kid first;
- Kid last;
- // 构造函数,构造n个孩子围成的圈
- public KidCircle( int n)
- {
- for ( int i = 0 ; i < n; i++)
- {
- add(); // 调用add方法,循环一次增加一个孩子
- }
- }
- // 添加孩子
- public void add()
- {
- Kid k = new Kid();
- k.id = count;
- if (count <= 0 )
- {
- first = k; // 只有一个孩子
- last = k;
- k.left = k;
- k.right = k;
- }
- else
- {
- last.right = k; // 从最后的节点开始,k成为last的右边孩子
- k.left = last; // last成为k的左边孩子
- k.right = first; // k成为first的右边孩子
- first.left = k; // k成为first的左边孩子
- last = k; // 这时候k变成了最后一个
- }
- count++;
- }
- // 删除孩子
- public void delete(Kid k)
- {
- if (count <= 0 ) // 没有孩子
- {
- return ;
- }
- else if (count == 1 ) // 只剩一个孩子
- {
- first = last = null ;
- }
- else
- {
- k.left.right = k.right; // k的右孩子成为 k的左孩子 的右边孩子
- k.right.left = k.left; // k的左孩子成为 k的右孩子 的左边孩子
- if (k == first) // 如果k是第一个孩子first
- {
- first = k.right;
- }
- else if (k == last) // 如果k是第一个孩子last
- {
- last = k.left;
- }
- }
- count--;
- }
- }