转载请注明出处:http://blog.csdn.net/ns_code/article/details/27957407

题目描写叙述:

每年六一儿童节,JOBDU都会准备一些小礼物去看望孤儿院的小朋友,今年亦是如此。

HF作为JOBDU的资深元老,自然也准备了一些小游戏。当中,有个游戏是这种:首先,让小朋友们围成一个大圈。

然后,他随机指定一个数m,让编号为1的小朋友開始报数。每次喊到m的那个小朋友要出列唱首歌,然后能够在礼品箱中随意的挑选礼物,而且不再回到圈中,从他的下一个小朋友開始,继续1...m报数....这样下去....直到剩下最后一个小朋友,能够不用表演,而且拿到JOBDU名贵的“名侦探柯南”典藏版(名额有限哦!!^_^)。

请你试着想下,哪个小朋友会得到这份礼品呢?

输入:

输入有多组数据。

每组数据一行,包括2个整数n(0<=n<=1,000,000),m(1<=m<=1,000,000),n,m分别表示小朋友的人数(编号1....n-1,n)和HF指定的那个数m(如上文所述)。

假设n=0,则结束输入。

输出:

相应每组数据,输出最后拿到大奖的小朋友编号。

例子输入:
1 10
8 5
6 6
0
例子输出:
1
3
4

约瑟夫环问题。最简单直观的方法就是用数组或者链表模拟整个游戏的过程,我先是用数组模拟实现了下。

数组模拟实现的代码例如以下:

#include<stdio.h>

typedef struct Node
{
int next; //下一个元素的编号
int num; //编号,从1開始
}Node; Node arr[1000000]; int FindLastRemaining(int n,int m)
{
if(n<1 || m<1)
return 0; int start = 1;
int current = start;
int pre = current;
while(start != arr[start].next)
{
//找到待删元素和它的前一个元素
int i;
for(i=0;i<m-1;i++)
{
pre = current;
current = arr[current].next;
} //删除元素,又一次建立连接
arr[pre].next = arr[current].next;
start = arr[pre].next;
current = start;
}
return arr[start].num;
} int main()
{
int n;
while(scanf("%d",&n) != EOF && n != 0)
{
int m;
scanf("%d",&m); //将数据连成环
int i;
for(i=1;i<=n;i++)
{
arr[i].num = i;
if(i == n)
arr[i].next = 1;
else
arr[i].next = i+1;
} printf("%d\n",FindLastRemaining(n,m));
}
return 0;
}

自己測试了几组数据,都对了,可是在九度OJ上測试,五个測试用例,仅仅通过两个,第二个測试用例WA。后面两个超时了。

watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvbW1jX21hb2R1bg==/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/SouthEast" alt="" />

超时就不说了,模拟实现的方法,时间复杂度为O(m*n)。数据非常大时时间消耗就上来了。可是第二个測试用例报了WA,左看右看还是不知道哪里出了问题,不想继续纠结下去了!看到这篇博文的有心者。有发现问题所在的话,提示下。

无奈这个思路的代码木有AC,即使结果正确,也会超时。仅仅能硬着头皮去研究下书上O(n)的做法,顺着书上面的思路。推了半个多小时。总算勉强搞定了,这种思路假设事先没个结论在胸中的话,面试现场基本是不可能推导出来的,除非曾经就知道并推导过这个数学公式。

关于思路。不说太多了。能够看剑指offer,上面写的还是蛮具体的,最好自己动手推推。这里重点说几点我的理解

1、个人感觉对推到时,最好不要用%来合并两个分段函数,最好p(x)依照x的范围分段,这样比較easy理解。这样p(x)有两段函数。求得的逆函数相同是分段函数。

2、k=(m-1)%n永远指的从0開始删除的第m个元素,也就是编号为m-1的元素。由于每次都会对元素的编号做映射,使每次从新開始报数的元素的编号都映射为从0開始。

3、书中通过找规律得到p(x)=(x-k-1)%n,这个感觉不严谨,由于x-k-有可能出现负的情况,应该写为p(x)=(n+x-k-1)%n。但终于求的的递推公式是一样的。由于这里不会出现超过2倍的情况。

另外,九度上的题目要求的编号是从1開始的,这简单。直接将结果的编号加1就可以。

AC代码:

#include<stdio.h>

int LastRemaining(int n,int m)
{
if(n<1 || m<1)
return 0; int last = 0;
int i;
for(i=2;i<=n;i++)
last = (last + m)%i;
return last;
} int main()
{
int n;
while(scanf("%d",&n) != EOF && n != 0)
{
int m;
scanf("%d",&m);
printf("%d\n",LastRemaining(n,m)+1);
}
return 0;
}
/**************************************************************
    Problem: 1356
    User: mmc_maodun
    Language: C
    Result: Accepted
    Time:290 ms
    Memory:912 kb
****************************************************************/

【剑指offer】约瑟夫环问题的更多相关文章

  1. C++版 - 剑指Offer 面试题45:圆圈中最后剩下的数字(约瑟夫环问题,ZOJ 1088:System Overload类似)题解

    剑指Offer 面试题45:圆圈中最后剩下的数字(约瑟夫环问题) 原书题目:0, 1, - , n-1 这n个数字排成一个圈圈,从数字0开始每次从圆圏里删除第m个数字.求出这个圈圈里剩下的最后一个数字 ...

  2. 剑指 Offer 62. 圆圈中最后剩下的数字 + 约瑟夫环问题

    剑指 Offer 62. 圆圈中最后剩下的数字 Offer_62 题目描述 方法一:使用链表模拟 这种方法是暴力方法,时间复杂度为O(nm),在本题中数据量过大会超时. 方法二:递归方法 packag ...

  3. 面试题目——《剑指Offer》

    1.把一个字符串转换成整数——<剑指Offer>P29 2.求链表中的倒数第k个结点——<剑指Offer>P30 3.实现Singleton模式——<剑指Offer> ...

  4. 【剑指Offer学习】【全部面试题汇总】

    剑指Offer学习 剑指Offer这本书已经学习完了.从中也学习到了不少的东西,如今做一个总的文件夹.供自已和大家一起參考.学如逆水行舟.不进则退.仅仅有不断地学习才干跟上时候.跟得上技术的潮流! 全 ...

  5. 《剑指offer》全部题目-含Java实现

    1.二维数组中的查找 在一个二维数组中,每一行都按照从左到右递增的顺序排序,每一列都按照从上到下递增的顺序排序.请完成一个函数,输入这样的一个二维数组和一个整数,判断数组中是否含有该整数. publi ...

  6. 【剑指Offer学习】【所有面试题汇总】

    剑指Offer学习 剑指Offer这本书已经学习完了,从中也学习到了不少的东西,现在做一个总的目录,供自已和大家一起参考,学如逆水行舟,不进则退.只有不断地学习才能跟上时候,跟得上技术的潮流! 所有代 ...

  7. 剑指offer题目java实现

    Problem2:实现Singleton模式 题目描述:设计一个类,我们只能生成该类的一个实例 package Problem2; public class SingletonClass { /* * ...

  8. 剑指offer算法总结

    剑指offer算法学习总结 节选剑指offer比较经典和巧妙的一些题目,以便复习使用.一部分题目给出了完整代码,一部分题目比较简单直接给出思路.但是不保证我说的思路都是正确的,个人对算法也不是特别在行 ...

  9. 剑指offer 面试62题

    面试62题: 题目:圆圈中最后剩下的数字 题:0,1,...,n-1这n个数字排成一个圆圈,从数字0开始,每次从这个圆圈里删除第m个数字.求出这个圆圈里剩下的最后一个数字. 解题思路:约瑟夫环问题,可 ...

随机推荐

  1. BLE简介和Android BLE编程

    一.BLE和BT区别 其实我知道许多程序员不太喜欢阅读除了代码以外的文档,因为有时这些过于冗长的文档对编程并没有更多的好处,有了协议,接口,demo差不多很多人就能写出很好质量的代码了.但其实更深入的 ...

  2. Delphi + Asm - TBits类的学习

    技术交流,DH讲解. 在D2010的classes中有个TBits类,这个类主要是位操作的. 1 2 3 4 5 6 7 8 9 10 11 12 13 14 TBits = class privat ...

  3. 配置VS2008下的Qt开发环境有感

    写一篇小小的日志为了在VS2008中安装Qt的插件,花了我很多的时间.1.vs2008在win7中破解问题我的VS2008已经安装好了,不知道为何,当初没有破解,现在只剩下15天限制了.于是为了破解, ...

  4. Delete it

    Problem A: Delete it Time Limit: 2 Sec   Memory Limit: 64 MB Submit: 99   Solved: 25 Description 克林在 ...

  5. [置顶] 64位Win2008_VS2012使用ODP.NET遭遇问题和解决办法

    最近为使用Oracle11G数据库做个快速开发的小程序,使用64位Win2008+Vs2012环境,结果碰壁连环,幸好不算太笨,终于解决了,特记录一下. 测试环境: Oracle11g (11.2.0 ...

  6. Android-Cannot merge new index 66195 into a non-jumbo instruction的解决的方法

    转载请注明来源:http://blog.csdn.net/goldenfish1919/article/details/33729679 用eclispe打包的时候报错: [2014-06-23 13 ...

  7. Delphi XE2 升级项目编译ralease版本,无法添加UAC解决方法

    我今天把一个原来是Delphi2007的工程升级到了Delphi XE2,在编译ralease版本时候,发现无法添加UAC,我觉得可能是升级的原因,随后我用XE2新建了一个工程,但是在编译raleas ...

  8. BestR #31

    hdu 5178 求|a[i] - a[j]| <= k (i < j) <i,j>的对数,一开始认为数据不大就直接ans++了,后来结果出来才知道,啊啊啊,too young ...

  9. Perl入门(四)Perl的正則表達式

    正則表達式是Perl语言的特色.主要的语法不是非常难,可是编写一个符合需求.高效的正則表達式.还是有一些挑战的. Perl的三种匹配模式 1.查找 语法:m/正則表達式内容/; 作用:查找匹配内容中是 ...

  10. alv 列标题

    gs_fieldcat-reptext_ddic才是显示列标题的