`

程序员面试题精选(14)-圆圈中最后剩下的数字

阅读更多

题目:n个数字(0,1,…,n-1)形成一个圆圈,从数字0开始,每次从这个圆圈中删除第m个数字(第一个为当前数字本身,第二个为当前数字的下一个数字)。当一个数字删除后,从被删除数字的下一个继续删除第m个数字。求出在这个圆圈中剩下的最后一个数字。
分析:既然题目有一个数字圆圈,很自然的想法是我们用一个数据结构来模拟这个圆圈。在常用的数据结构中,我们很容易想到用环形列表。我们可以创建一个总共有m个数字的环形列表,然后每次从这个列表中删除第m个元素。
在参考代码中,我们用STL中std::list来模拟这个环形列表。由于list并不是一个环形的结构,因此每次跌代器扫描到列表末尾的时候,要记得把跌代器移到列表的头部。这样就是按照一个圆圈的顺序来遍历这个列表了。
这种思路需要一个有n个结点的环形列表来模拟这个删除的过程,因此内存开销为O(n)。而且这种方法每删除一个数字需要m步运算,总共有n个数字,因此总的时间复杂度是O(mn)。当m和n都很大的时候,这种方法是很慢的。
接下来我们试着从数学上分析出一些规律。首先定义最初的n个数字(0,1,…,n-1)中最后剩下的数字是关于n和m的方程为f(n,m)。
在这n个数字中,第一个被删除的数字是m%n-1,为简单起见记为k。那么删除k之后的剩下n-1的数字为0,1,…,k-1,k+1,…,n-1,并且下一个开始计数的数字是k+1。相当于在剩下的序列中,k+1排到最前面,从而形成序列k+1,…,n-1,0,…k-1。该序列最后剩下的数字也应该是关于n和m的函数。由于这个序列的规律和前面最初的序列不一样(最初的序列是从0开始的连续序列),因此该函数不同于前面函数,记为f’(n-1,m)。最初序列最后剩下的数字f(n,m)一定是剩下序列的最后剩下数字f’(n-1,m),所以f(n,m)=f’(n-1,m)。
接下来我们把剩下的的这n-1个数字的序列k+1,…,n-1,0,…k-1作一个映射,映射的结果是形成一个从0到n-2的序列:
k+1     ->     0
k+2     ->     1

n-1     ->     n-k-2
0    ->     n-k-1

k-1    ->    n-2
把映射定义为p,则p(x)= (x-k-1)%n,即如果映射前的数字是x,则映射后的数字是(x-k-1)%n。对应的逆映射是p-1(x)=(x+k+1)%n。
由于映射之后的序列和最初的序列有同样的形式,都是从0开始的连续序列,因此仍然可以用函数f来表示,记为f(n-1,m)。根据我们的映射规则,映射之前的序列最后剩下的数字f’(n-1,m)= p-1 [f(n-1,m)]=[f(n-1,m)+k+1]%n。把k=m%n-1代入得到f(n,m)=f’(n-1,m)=[f(n-1,m)+m]%n。
经过上面复杂的分析,我们终于找到一个递归的公式。要得到n个数字的序列的最后剩下的数字,只需要得到n-1个数字的序列的最后剩下的数字,并可以依此类推。当n=1时,也就是序列中开始只有一个数字0,那么很显然最后剩下的数字就是0。我们把这种关系表示为:
          0                   n=1
f(n,m)={
          [f(n-1,m)+m]%n      n>1
尽管得到这个公式的分析过程非常复杂,但它用递归或者循环都很容易实现。最重要的是,这是一种时间复杂度为O(n),空间复杂度为O(1)的方法,因此无论在时间上还是空间上都优于前面的思路。
思路一的参考代码:

C++代码 复制代码
  1. ///////////////////////////////////////////////////////////////////////   
  2. // n integers (0, 1, ... n - 1) form a circle. Remove the mth from    
  3. // the circle at every time. Find the last number remaining    
  4. // Input: n - the number of integers in the circle initially   
  5. //         m - remove the mth number at every time   
  6. // Output: the last number remaining when the input is valid,   
  7. //          otherwise -1   
  8. ///////////////////////////////////////////////////////////////////////   
  9. int LastRemaining_Solution1(unsigned int n, unsigned int m)   
  10. {   
  11.       // invalid input   
  12.       if(n < 1 || m < 1)   
  13.             return -1;   
  14.       unsigned int i = 0;   
  15.       // initiate a list with n integers (0, 1, ... n - 1)   
  16.        list<int> integers;   
  17.       for(i = 0; i < n; ++ i)   
  18.              integers.push_back(i);   
  19.        list<int>::iterator curinteger = integers.begin();   
  20.       while(integers.size() > 1)   
  21.        {   
  22.             // find the mth integer. Note that std::list is not a circle   
  23.             // so we should handle it manually   
  24.             for(int i = 1; i < m; ++ i)   
  25.              {   
  26.                    curinteger ++;   
  27.                   if(curinteger == integers.end())   
  28.                          curinteger = integers.begin();   
  29.              }   
  30.   
  31.             // remove the mth integer. Note that std::list is not a circle   
  32.             // so we should handle it manually   
  33.              list<int>::iterator nextinteger = ++ curinteger;   
  34.             if(nextinteger == integers.end())   
  35.                    nextinteger = integers.begin();   
  36.              -- curinteger;   
  37.              integers.erase(curinteger);   
  38.              curinteger = nextinteger;   
  39.        }   
  40.   
  41.       return *(curinteger);   
  42. }  
///////////////////////////////////////////////////////////////////////
// n integers (0, 1, ... n - 1) form a circle. Remove the mth from 
// the circle at every time. Find the last number remaining 
// Input: n - the number of integers in the circle initially
//         m - remove the mth number at every time
// Output: the last number remaining when the input is valid,
//          otherwise -1
///////////////////////////////////////////////////////////////////////
int LastRemaining_Solution1(unsigned int n, unsigned int m)
{
      // invalid input
      if(n < 1 || m < 1)
            return -1;
      unsigned int i = 0;
      // initiate a list with n integers (0, 1, ... n - 1)
       list<int> integers;
      for(i = 0; i < n; ++ i)
             integers.push_back(i);
       list<int>::iterator curinteger = integers.begin();
      while(integers.size() > 1)
       {
            // find the mth integer. Note that std::list is not a circle
            // so we should handle it manually
            for(int i = 1; i < m; ++ i)
             {
                   curinteger ++;
                  if(curinteger == integers.end())
                         curinteger = integers.begin();
             }

            // remove the mth integer. Note that std::list is not a circle
            // so we should handle it manually
             list<int>::iterator nextinteger = ++ curinteger;
            if(nextinteger == integers.end())
                   nextinteger = integers.begin();
             -- curinteger;
             integers.erase(curinteger);
             curinteger = nextinteger;
       }

      return *(curinteger);
}



思路二的参考代码:

C++代码 复制代码
  1. ///////////////////////////////////////////////////////////////////////   
  2. // n integers (0, 1, ... n - 1) form a circle. Remove the mth from    
  3. // the circle at every time. Find the last number remaining    
  4. // Input: n - the number of integers in the circle initially   
  5. //         m - remove the mth number at every time   
  6. // Output: the last number remaining when the input is valid,   
  7. //          otherwise -1   
  8. ///////////////////////////////////////////////////////////////////////   
  9. int LastRemaining_Solution2(int n, unsigned int m)   
  10. {   
  11.       // invalid input   
  12.       if(n <= 0 || m < 0)   
  13.             return -1;   
  14.   
  15.       // if there are only one integer in the circle initially,   
  16.       // of course the last remaining one is 0   
  17.       int lastinteger = 0;   
  18.   
  19.       // find the last remaining one in the circle with n integers   
  20.       for (int i = 2; i <= n; i ++)    
  21.              lastinteger = (lastinteger + m) % i;   
  22.   
  23.       return lastinteger;   
  24. }  

分享到:
评论

相关推荐

    程序员面试题精选100题-何海涛

    《程序员面试题精选100题—何海涛》是一份详实的IT面试准备资料,由何海涛整理发布,旨在帮助应届毕业生和求职者准备面向微软、谷歌等知名科技公司的面试。此资料不仅收录了精选的100道面试题目,还提供了详细的解题...

    java程序员面试题150例-java常见面试题-java工程师面试题-java面试题大全

    java程序员面试题150例 java常见面试题 java工程师面试题 java面试题大全 带搜索功能,能非常方便的查找到你想要了解的 java面试题目 推荐大家下载。

    程序员面试题精选100题

    程序员面试题精选100题 本资源是程序员面试题精选100题,涵盖了算法、数据结构、操作系统、计算机网络、数据库等多个领域。今天,我们将深入分析其中的一道题目,即将二元查找树转换成排序的双向链表。 知识点一:...

    简历模板-程序员-通用-精选

    简历模板-程序员-通用-精选简历模板-程序员-通用-精选简历模板-程序员-通用-精选简历模板-程序员-通用-精选简历模板-程序员-通用-精选简历模板-程序员-通用-精选简历模板-程序员-通用-精选简历模板-程序员-通用-精选...

    java程序员早期面试题汇总.zip

    ------------------------------------- java程序员早期面试题汇总 BAT经典面试题汇总.pdf Java常考面试题.pdf java面试题(题库全)....程序员面试题精选100题.pdf ... -------------------------------------

    程序员考试试题---程序员考试教程

    在准备程序员考试的过程中,考生应当重视这些试题的分析和解答,尤其要关注历年常考的知识点,如面向对象编程的基本概念,常用数据结构(如数组、链表、树、图)的操作,基本算法(排序、搜索)的实现,以及软件设计...

    程序员面试宝典 第5版-欧立奇.part2.rar

    程序员面试宝典 第5版-欧立奇.part2.rar

    程序员面试题精选100题(经典!)

    具体来说,文档中详细讨论了一个在程序员面试中常见的算法问题:如何将二元查找树(Binary Search Tree,BST)转换为一个排序的双向链表。这是一个涉及树的遍历、指针操作、递归思维的经典算法问题。下面将详细介绍...

    程序员面试题精选(1-57)

    通过解决"程序员面试题精选(1-57)"中的题目,你可以系统地复习和巩固这些知识点,同时也可以训练自己面对新问题时的分析和解决问题的能力。这个资源不仅适用于面试准备,也是提升自身编程素养的好材料。因此,无论你...

    人事面试题150问-人事面试题大全-net程序员面试题--软件开发工程师面试题.xls

    人事面试题150问完整版,附加.NET程序员面试题,所有题目绝对真实有效,欢迎下载!O(∩_∩)O哈哈~

    程序员面试题精选100题【数据结构 /算法】

    【程序员面试题精选100题【数据结构 /算法】】是针对求职程序员精心挑选的一系列面试题目,涵盖了数据结构和算法两大核心领域。这些题目旨在帮助应聘者提高面试技巧,提升对技术的理解,以便在激烈的就业市场竞争中...

    程序员面试手册-超级全的程序员面试题-CHM版

    Java面试题,J2EE面试题,.net面试题,PHP面试题,数据库面试题,英语面试,外企面试,软件测试面试题,Python面试题,Oracle面试题,MySql面试题,Web开发面试题,Unix面试题,程序员面试,网络技术面试题,网络安全面试题,Linux...

    程序员面试宝典 第5版-欧立奇.part1.rar

    程序员面试宝典 第5版-欧立奇.part1.rar

    程序员面试宝典 第5版-欧立奇

    《程序员面试宝典 第5版-欧立奇》是一本为准备程序员面试而精心编写的指南,涵盖了软件开发领域中的核心知识和技能。这本书旨在帮助求职者在竞争激烈的IT行业中脱颖而出,通过深入理解和掌握关键概念,提升自己的...

    程序员面试题精选100题完整版

    在软件开发行业中,数据结构与算法是程序员必须掌握的基础技能之一,尤其是在求职过程中,这部分知识更是考察的重点。其中,二叉查找树(Binary Search Tree, BST)作为一种重要的数据结构,因其高效的查找特性而在...

    IT程序员面试题目及答案-计算机面试.docx

    IT程序员面试题目及答案-计算机面试.docx

Global site tag (gtag.js) - Google Analytics