题目描述:有n个人编号1-n,按照顺时针方向围成一个圆圈。它们预先定义好两个整数x,y。先从1号顺时针方向开始报数,报到x的人出圈,再从x的逆时针方向的后一个人从1开始报数,报到y的人出圈,再从这个人的顺时针方向后一个人开始从1报数,报到x的人出圈,如此反复,直到最后剩下一个人为止,问最后剩下的那个人是几号? 比如n = 10, x = 3, y = 2,报数的过程如下 报数人 1, 2, 3 3出圈 报数人 2,1 1出圈 接着2,4 ,5 5出圈 接着4,2 2出圈 接着4,6,7 7出圈 接着6,4 4出圈 接着6,8,9 9出圈 接着8,6 6出圈 接着8,10,8 8出圈 剩余 10号。 输入n,x,y输出剩余的编号。 数据范围 1 < n <= 1000000, 1<=x,y<=1000000000。
我的代码如下:
思想是,先建立一个链表,每次按要求从中删除一个,当然这样可以很确定删除的是哪一个,最后剩下的那个就是所求结果。
using System;using System.Collections.Generic;public class Test{public static int remain(int n,int x,int y){if(n==1)return 1;List<int> ln = new List<int> ();int i,j;for(i=0;i<n;i++) ln.Add(i+1);j=0;int ti=0;while(n>1){j=j+1;if(j==1){ti=(ti+x-1)%n;//Console.WriteLine("第{0}个,数字为{1}",ti,ln[ti]);ln.RemoveAt(ti);n=n-1;if(ti>0) ti--;else ti = n-1;}else{j=j%2;ti=(ti-(y-1))%n;if(ti<0) ti+=n;//Console.WriteLine("第{0}个,数字为{1}",ti,ln[ti]);//Console.WriteLine(ln[ti]);ln.RemoveAt(ti);n--;if(ti>=n) ti=ti-n;}}return ln[0];}
我本来以为这样耗时也就是O(n),当时没有想到对List进行操作是很费时间的,根据下标访问某个元素的耗时也是O(n),因此上述算法就变成了O(n^2),因此提交没有通过。
看到某高手的文章,才恍然大悟,原来还可以这样!