题目描述:

  每年六一儿童节,牛客都会准备一些小礼物去看望孤儿院的小朋友,今年亦是如此。HF作为牛客的资深元老,自然也准备了一些小游戏。其中,有个游戏是这样的:首先,让小朋友们围成一个大圈。然后,他随机指定一个数m,让编号为0的小朋友开始报数。每次喊到m-1的那个小朋友要出列唱首歌,然后可以在礼品箱中任意的挑选礼物,并且不再回到圈中,从他的下一个小朋友开始,继续0...m-1报数....这样下去....直到剩下最后一个小朋友,可以不用表演,并且拿到牛客名贵的“名侦探柯南”典藏版(名额有限哦!!^_^)。请你试着想下,哪个小朋友会得到这份礼品呢?(注:小朋友的编号是从0到n-1)。

  解题思路:

  本题就是著名的约瑟夫(Josephuse)环的问题,对于这个有名的算法问题,程序员应该不陌生,这里我们给出两种比较经典的解法。

  方法一:用环形链表模拟圆圈。这实际上是一个比较直观的解法,既然题目中有一个数字圆圈,用一个数据结构来模拟游戏过程是一个比较自然的想法。而在数据结构中,比较合适的是用环形链表,因为涉及到结点的删除和顺序进行查找的操作。

  方法二:递推公式法。

  我们知道第一个人(编号一定是m%n-1) 出列之后,剩下的n-1个人组成了一个新的约瑟夫环(以编号为k=m%n的人开始):

  k k+1 k+2 ... n-2, n-1, 0, 1, 2, ... k-2并且从k开始报0。

  现在我们把他们的编号做一下转换:

  k --> 0

  k+1 --> 1

  k+2 --> 2

  ...

  ...

  k-2 --> n-2

  k-1 --> n-1

  变换后就完完全全成为了(n-1)个人报数的子问题,假如我们知道这个子问题的解:

  例如x是最终的胜利者,那么根据上面这个表把这个x变回去不刚好就是n个人情况的解吗?

  变回去的公式很简单,相信大家都可以推出来:x'=(x+k)%n。

  令f[i]表示i个人玩游戏报m退出最后胜利者的编号,最后的结果自然是f[n]。

  递推公式:

  f[1]=0;

  f[i]=(f[i-1]+m)%i; (i>1)

  举例:


  编程实现(Java):

public class Solution {public int LastRemaining_Solution(int n, int m) {/*著名的约瑟夫环问题f(n,m)=(f(n-1,m)+m)%n;*/if(n<1 || m<1)return -1;int last=0; //n=1时为0for(int i=2;i<=n;i++){last=(last+m)% i;}return last;}
}

转载于:https://www.cnblogs.com/gzshan/p/10856550.html

【剑指Offer】46、圆圈中最后剩下的数相关推荐

  1. 剑指offer:圆圈中最后剩下的数

    题目描述 每年六一儿童节,牛客都会准备一些小礼物去看望孤儿院的小朋友,今年亦是如此.HF作为牛客的资深元老,自然也准备了一些小游戏.其中,有个游戏是这样的:首先,让小朋友们围成一个大圈.然后,他随机指 ...

  2. 【LeetCode】剑指 Offer 62. 圆圈中最后剩下的数字

    [LeetCode]剑指 Offer 62. 圆圈中最后剩下的数字 文章目录 [LeetCode]剑指 Offer 62. 圆圈中最后剩下的数字 一.动态规划 总结 一.动态规划 构建一个长度为 n ...

  3. java求最后一位不为0的数字_【Java】 剑指offer(62) 圆圈中最后剩下的数字

    本文参考自<剑指offer>一书,代码采用Java语言. 题目 0, 1, -, n-1这n个数字排成一个圆圈,从数字0开始每次从这个圆圈里删除第m个数字.求出这个圆圈里剩下的最后一个数字 ...

  4. 90. Leetcode 剑指 Offer 62. 圆圈中最后剩下的数字 (动态规划-基础题)

    步骤一.确定状态: 确定dp数组及含义 dp[i]表示的是i个数里面循环删除m个数剩下的数 步骤二.推断状态方程: 首先假设f(n,m)表示从n个数字0....n−1中每次删除第 m个数字之后剩下的数 ...

  5. 【LeetCode笔记】剑指 Offer 62. 圆圈中最后剩下的数字(Java、约瑟夫环、链表)

    文章目录 题目描述 思路 & 代码 链表模拟法 数学方法 二刷 题目描述 约瑟夫环!题目可太经典了 说实话还是有点难度的= = 思路 & 代码 链表模拟法 第一想法是用 LinkedL ...

  6. 打卡系列-剑指 Offer 62. 圆圈中最后剩下的数字

    0,1,,n-1这n个数字排成一个圆圈,从数字0开始,每次从这个圆圈里删除第m个数字.求出这个圆圈里剩下的最后一个数字. 例如,0.1.2.3.4这5个数字组成一个圆圈,从数字0开始每次删除第3个数字 ...

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

    class Solution {// f(n, m) = (f(n – 1, m) + m) % npublic int lastRemaining(int n, int m) {return (n ...

  8. 《剑指offer》-- 栈的压入与弹出序列、把字符串转化为整数、扑克牌顺子、孩子们的游戏(圆圈中最后剩下的数)

    一.栈的压入与弹出序列: 1.题目: 输入两个整数序列,第一个序列表示栈的压入顺序,请判断第二个序列是否可能为该栈的弹出顺序. 假设压入栈的所有数字均不相等.例如序列1,2,3,4,5是某栈的压入顺序 ...

  9. leetcode 剑指 Offer 46. 把数字翻译成字符串

    剑指 Offer 46. 把数字翻译成字符串 给定一个数字,我们按照如下规则把它翻译为字符串:0 翻译成 "a" ,1 翻译成 "b",--,11 翻译成 &q ...

  10. 【每日一题】剑指 Offer 22. 链表中倒数第k个节点

    剑指 Offer 22. 链表中倒数第k个节点

最新文章

  1. django教程目录
  2. python简单的小程序_初学python的一些简单程序(1)
  3. 隐函数画图with R
  4. JSP连接SQL数据库实现数据分页显示
  5. python 多线程读写文件错误_python多线程老是报错。大神帮忙看看哈?
  6. linux的文件系统架构
  7. easyui-combobox 模糊匹配 支持汉字和拼音_巧用数据验证制作模糊匹配的下拉列表...
  8. C#图片上一张下一张
  9. 计算机无法共享打印机共享的打印机驱动,打印机不能共享_打印机不能共享怎么办?-太平洋IT百科...
  10. Django——jinja2配置与使用
  11. encode()和decode()
  12. python拼多多推广多店爬虫
  13. linux c 获取进程p id,详解Linux获取线程的PID(TID、LWP)的几种方式
  14. eclipse折叠if语句块_「03」java中的方法以及控制语句
  15. Linux增加硬盘详细教程,Linux 添加新硬盘
  16. oracle 考勤记录表,基于C#+Oracle的考勤管理系统的设计与开发_.doc
  17. 百度云盘archlinux manjaro直接安装
  18. 电容笔哪个品牌好?十大电容笔知名品牌
  19. Audio的谐波失真
  20. 还没35岁的我已经快秃了

热门文章

  1. django开发-在Docker中部署django项目
  2. Kickstarter 开源其 Android 和 iOS 应用
  3. SAP WM 工单完工入库,系统报错- No SU type could be determined -
  4. RedHat 7配置bonding双网卡绑定
  5. 通过php jq ajax 提交form表单
  6. cPanel虚拟主机上运行Python的方法
  7. 二级域名会不会分散主域名权重
  8. JFinal自动扫描表绑定model(包含jar包扫描)
  9. TCP/IP协议栈的封装
  10. vscode 设置指南