剑指offer--孩子们的游戏(圆圈中最后剩下的数字)
每年六一儿童节,牛客都会准备一些小礼物去看望孤儿院的小朋友,今年亦是如此。HF作为牛客的资深元老,自然也准备了一些小游戏。其中,有个游戏是这样的:首先,让小朋友们围成一个大圈。然后,他随机指定一个数m,让编号为0的小朋友开始报数。每次喊到m-1的那个小朋友要出列唱首歌,然后可以在礼品箱中任意的挑选礼物,并且不再回到圈中,从他的下一个小朋友开始,继续0...m-1报数....这样下去....直到剩下最后一个小朋友,可以不用表演,并且拿到牛客名贵的“名侦探柯南”典藏版(名额有限哦!!^_^)。请你试着想下,哪个小朋友会得到这份礼品呢?(注:小朋友的编号是从0到n-1)
解法一:数组(python实现)
维持一个数组,和一个索引cur指向要删除的位置,当cur的指等于数组的大小时,令cur回到头位置
class Solution:
def LastRemaining_Solution(self, n, m):
if n<1 or m < 1: return -1
children = list(range(n))
cur = -1
while len(children) > 1:
for i in range(m):
cur += 1
if(cur == len(children)):
cur = 0
del children[cur]
//在新的list中cur指向了下一个元素,为了保证移动m个的准确性,cur需要向前移动一位
cur -= 1
return children[0]
解法二:循环链表(c++实现)
构建一个循环链表依次删除喊m-1的节点,当链表中只剩一个节点时,输出这个节点的值
class Solution {
public:
int LastRemaining_Solution(int n, int m)
{
if(n<1||m<1) return -1;
ListNode *pHead = createCircularListNode(n);
ListNode *pNode = pHead;
while(pNode!=pNode->next){
//将喊m-1的孩子移出圆圈
for(int j = 1; j<m-1; j++){
pNode = pNode->next;
}
ListNode *ptemp = pNode->next;
pNode->next = ptemp->next;
free(ptemp);
pNode = pNode->next;
}
return pNode->val;
}
// 创建循环链表
ListNode *createCircularListNode(int n){
ListNode *phead, *pNode;
phead = NULL;
//创建链表
if(n!=0){
for(int i = 0; i < n; i++){
ListNode* ptemp = (ListNode*)malloc(sizeof(ListNode));
ptemp->val = i;
if(phead == NULL){
phead = ptemp;
pNode = phead;
}
else{
pNode->next = ptemp;
pNode = ptemp;
}
}
//将尾指针指向头指针
pNode->next = phead;
}
return phead;
}
};
时间复杂度O(n) ,空间复杂度O(n)
解法三:数学公式(JAVA实现)
对于n个孩子参与的游戏,第一轮游戏删除带有下划线的那个(喊m-1的孩子) 0, 1, 2, .... m-1, m, m+1, m+2, .... n-2, n-1
得到新的数组 0, 1, 2, .... m-2, m, m+1, m+2, .... n-2, n-1
从m处重新开始从0报数,上述数组重新排列 m, m+1, m+2, .... n-2, n-1,0, 1, 2, ...., m-3, m-2.
按照上边的顺序从0开始编号,对应的编号为 0, 1, 2,... n-(m+2), n-(m+1), n-m, m-(m-1), n-(m-2), ..., n-3, n-2
若上一行为x' 下一行为x,则对应关系为:x'= (x+m) % n
通过上表可得,将1人出队后的数据重新组织成0-(n-2) 共计n-1个人的列表,并求由n-1个人参与,并将其中报m-1的人出列的问题.也就是说要求原问题n个人参与的解,可以先求n-1个人参与的解,然后通过转换公式得出n个人参与的解.
因此当n=1时是规模最小的情况,F(1) = 0, 当n=2时根据公式可知问题的解F(2) = (F(1)+m)%2, 当有n个人时问题的解 F(n) = (F(n-1)+m)%n, 可以用递归也可以用递推的方式解决这个问题,如果用递归解决的话存在大量重复计算的问题,因此我们可以用递推的方式求解
public class Solution {
public int LastRemaining_Solution(int n, int m) {
if(n<1||m<1) return -1;
int cur=0;
for(int i = 1; i < n; i++){
cur = (cur+m)%(i+1);
}
return cur;
}
}
时间复杂度O(n) ,空间复杂度O(1)
剑指offer--孩子们的游戏(圆圈中最后剩下的数字)的更多相关文章
- 剑指Offer——孩子们的游戏(圆圈中最后剩下的数)
题目描述: 每年六一儿童节,牛客都会准备一些小礼物去看望孤儿院的小朋友,今年亦是如此.HF作为牛客的资深元老,自然也准备了一些小游戏.其中,有个游戏是这样的:首先,让小朋友们围成一个大圈.然后,他随机 ...
- 剑指offer--42.孩子们的游戏(圆圈中最后剩下的数)
约瑟夫环,用链表,队列,总之模拟过程 ----------------------------------------------------------------- 时间限制:1秒 空间限制:32 ...
- 剑指Offer-46.孩子们的游戏(圆圈中最后剩下的数)(C++/Java)
题目: 每年六一儿童节,牛客都会准备一些小礼物去看望孤儿院的小朋友,今年亦是如此.HF作为牛客的资深元老,自然也准备了一些小游戏.其中,有个游戏是这样的:首先,让小朋友们围成一个大圈.然后,他随机指定 ...
- 剑指 offer set 21 圆圈中最后剩下的数字
思路 1. 经典解法是用环形链表模拟圆圈, 然后每次减少一个节点. 时间复杂度为 o(mn), 空间复杂度为 o(n) 2. 转化成数学问题, 递推公式决定下一个元素. 时间复杂度为 o(n), 空间 ...
- 剑指Offer - 九度1356 - 孩子们的游戏(圆圈中最后剩下的数)
剑指Offer - 九度1356 - 孩子们的游戏(圆圈中最后剩下的数)2014-02-05 19:37 题目描述: 每年六一儿童节,JOBDU都会准备一些小礼物去看望孤儿院的小朋友,今年亦是如此.H ...
- 【剑指Offer】孩子们的游戏(圆圈中最后剩下的数) 解题报告(Python)
[剑指Offer]孩子们的游戏(圆圈中最后剩下的数) 解题报告(Python) 标签(空格分隔): 剑指Offer 题目地址:https://www.nowcoder.com/ta/coding-in ...
- 【剑指Offer面试编程题】题目1356:孩子们的游戏(圆圈中最后剩下的数)--九度OJ
题目描述: 每年六一儿童节,JOBDU都会准备一些小礼物去看望孤儿院的小朋友,今年亦是如此.HF作为JOBDU的资深元老,自然也准备了一些小游戏.其中,有个游戏是这样的:首先,让小朋友们围成一个大圈. ...
- 【剑指offer】50.数组中重复出现的数字
50.数组中重复出现的数字 知识点:数组:Set的不可重复性 题目描述 在一个长度为n的数组里的所有数字都在0到n-1的范围内. 数组中某些数字是重复的,但不知道有几个数字是重复的.也不知道每个数字重 ...
- 剑指Offer 46. 孩子们的游戏(圆圈中最后剩下的数) (其他)
题目描述 每年六一儿童节,牛客都会准备一些小礼物去看望孤儿院的小朋友,今年亦是如此.HF作为牛客的资深元老,自然也准备了一些小游戏.其中,有个游戏是这样的:首先,让小朋友们围成一个大圈.然后,他随机指 ...
- 剑指offer四十六之孩子们的游戏(圆圈中最后剩下的数,约瑟夫环问题)
一.题目 每年六一儿童节,牛客都会准备一些小礼物去看望孤儿院的小朋友,今年亦是如此.HF作为牛客的资深元老,自然也准备了一些小游戏.其中,有个游戏是这样的:首先,让小朋友们围成一个大圈.然后,他随机指 ...
随机推荐
- Flutter: random color
import 'dart:math' as math; import 'package:flutter/material.dart'; void main() => runApp(App()); ...
- Mybatis-04 日志、分页
Mybatis-04 日志.分页 日志 1.日志工厂 如果数据库操作出现异常,就需要打印日志来排错. 日志工厂会把日志工作委托实现: SLF4J Apache Commons Logging Log4 ...
- 分分钟钟学会Python - 第四章 文件操作
4.1 文件基本操作 obj = open('路径',mode='模式',encoding='编码') obj.write() # 写入 obj.read() # 读取 obj.close() #关闭 ...
- QuickBase64 - Android 下拉通知栏快捷base64加解密工具
Android Quick Setting Tile Base64 Encode/Decode Tool Android 下拉通知栏快捷 base64 加解密,自动将剪切板的内容进行 base64 E ...
- STL中常用容器及操作 学习笔记1
@[TOC](下面介绍STL中常见的容器及操作)## 不定长数组 vector> vetcor:其实就是一个数组或者说是容器 其操作不同于之前直接定义的数组 > 而且可以直接赋值也可以直接 ...
- mysql查询缓存简单使用
MySQL是一个关系型数据库管理系统,由瑞典MySQL AB 公司开发,属于 Oracle 旗下产品.MySQL 是最流行的关系型数据库管理系统之一,在 WEB 应用方面,MySQL是最好的 RDBM ...
- 用c++解一元二次方程
解方程 github项目地址 这两天得知初二的表妹学了一元二次方程,听说还不会解,我就想着试试用C语言编写解方程. 一元二次方程 用公式法 这种方法效果很好: #include"funct. ...
- HTML基础速览
HTML概述 HTML ,CSS , JavaScript, JQuery, Vue 的关系 HTML可以写一个简单的前端,但是很丑,所以需要CSS对HTML进行美化 HTML是静态的.JavaScr ...
- Java基础:常用基础dos命令
打开cmd的方式1.开始+系统+命令提示符2.win键+R 输入cmd 打开控制台3.在任意的文件夹下,按住shift键+鼠标右键点击,在此处打开命令提示行4.在资源管理器的地址栏前面加上cmd路径 ...
- 自定义校验注解ConstraintValidator
一 前言 系统执行业务逻辑之前,会对输入数据进行校验,检测数据是否有效合法的.所以我们可能会写大量的if else等判断逻辑,特别是在不同方法出现相同的数据时,校验的逻辑代码会反复出现,导致代码冗余, ...