转载请注明出处: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. SharePoint 2013 &quot;通知我&quot;简单的功能

    简单的功能 "通知我"内部列表或文档库中的主要项目.加入/删除/修改等操作,用户的E- mail通知设定功能:设置列表或文档库通知的能力,有可能设置通知为一个单一的项目.这是Sha ...

  2. GitHub详解(转)

    GitHub 是一个共享虚拟主机服务,用于存放使用Git版本控制的软件代码和内容项目.它由GitHub公司(曾称Logical Awesome)的开发者Chris Wanstrath.PJ Hyett ...

  3. Eclipse和PyDev搭建完美Python开发环境(Windows篇)(转)

      摘要:本文讲解了用Eclipse和PyDev搭建Python的开发环境. 十一长假在家闲着没事儿,准备花点时间学习一下Python. 今儿花了一个下午搭建Python的开发环境,不禁感叹————开 ...

  4. poj 2777 Count Color(线段树区区+染色问题)

    题目链接:  poj 2777 Count Color 题目大意:  给出一块长度为n的板,区间范围[1,n],和m种染料 k次操作,C  a  b  c 把区间[a,b]涂为c色,P  a  b 查 ...

  5. Android中ListView.getCount()与ListView.getChildCount()区别和OnScrollListener()各个参数的区别

    istView.getCount()(实际上是 AdapterView.getCount()) 返回的是其 Adapter.getCount() 返回的值.也就是“所包含的 Item 总个数”. Li ...

  6. HTTPS的学习

    HTTPS的学习总结   HTTPS学习总结 简述 HTTPS对比HTTP就多了一个安全层SSL/TLS,具体就是验证服务端的证书和对内容进行加密. 先来看看HTTP和HTTPS的区别 我用AFN访问 ...

  7. [置顶] Java字节码文件剖析

    Java为什么能够支持跨平台,其实关键就是在于其*.class字节码文件,因为*.class字节码文件有一个统一标准的规范,里面是JVM运行的时需要的相关指令,各家的JVM必须能够解释编译执行标准字节 ...

  8. Valera and Tubes

    C. Valera and Tubes time limit per test 1 second memory limit per test 256 megabytes input standard ...

  9. STL algorithm算法lower_bound和upper_bound(31)

    lower_bound原型: function template <algorithm> std::lower_bound default (1) template <class F ...

  10. 【deep learning学习笔记】注释yusugomori的LR代码 --- LogisticRegression.h

    继续看yusugomori的代码,看逻辑回归.在DBN(Deep Blief Network)中,下面几层是RBM,最上层就是LR了.关于回归.二类回归.以及逻辑回归,资料就是前面转的几篇.套路就是设 ...