每年六一儿童节,牛客都会准备一些小礼物去看望孤儿院的小朋友,今年亦是如此。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--孩子们的游戏(圆圈中最后剩下的数字)的更多相关文章

  1. 剑指Offer——孩子们的游戏(圆圈中最后剩下的数)

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

  2. 剑指offer--42.孩子们的游戏(圆圈中最后剩下的数)

    约瑟夫环,用链表,队列,总之模拟过程 ----------------------------------------------------------------- 时间限制:1秒 空间限制:32 ...

  3. 剑指Offer-46.孩子们的游戏(圆圈中最后剩下的数)(C++/Java)

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

  4. 剑指 offer set 21 圆圈中最后剩下的数字

    思路 1. 经典解法是用环形链表模拟圆圈, 然后每次减少一个节点. 时间复杂度为 o(mn), 空间复杂度为 o(n) 2. 转化成数学问题, 递推公式决定下一个元素. 时间复杂度为 o(n), 空间 ...

  5. 剑指Offer - 九度1356 - 孩子们的游戏(圆圈中最后剩下的数)

    剑指Offer - 九度1356 - 孩子们的游戏(圆圈中最后剩下的数)2014-02-05 19:37 题目描述: 每年六一儿童节,JOBDU都会准备一些小礼物去看望孤儿院的小朋友,今年亦是如此.H ...

  6. 【剑指Offer】孩子们的游戏(圆圈中最后剩下的数) 解题报告(Python)

    [剑指Offer]孩子们的游戏(圆圈中最后剩下的数) 解题报告(Python) 标签(空格分隔): 剑指Offer 题目地址:https://www.nowcoder.com/ta/coding-in ...

  7. 【剑指Offer面试编程题】题目1356:孩子们的游戏(圆圈中最后剩下的数)--九度OJ

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

  8. 【剑指offer】50.数组中重复出现的数字

    50.数组中重复出现的数字 知识点:数组:Set的不可重复性 题目描述 在一个长度为n的数组里的所有数字都在0到n-1的范围内. 数组中某些数字是重复的,但不知道有几个数字是重复的.也不知道每个数字重 ...

  9. 剑指Offer 46. 孩子们的游戏(圆圈中最后剩下的数) (其他)

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

  10. 剑指offer四十六之孩子们的游戏(圆圈中最后剩下的数,约瑟夫环问题)

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

随机推荐

  1. 星空值、SPC、算力组成三元永动机制!VAST带你把握时代!

    目前中心化金融体系为用户提供的服务在便捷性和易用性方面已经达到了新高度,但随着时代发展,大众对于金融安全性和可控性的需求进一步提升,需要去中心化金融服务商来提供更具创意的解决方案.盛大公链为此在应用层 ...

  2. Java中print、printf、println的区别

    Java中print.printf.println的区别 区别 print:标准输出,但不换行,不可以空参: println:标准输出,但会自动换行,可以空参,可以看做:println()相当于pri ...

  3. 关于Python 编码的一点认识

    在计算机中,所有数据的存储.运算以及传输都必须是二进制数字,因为计算机只认识0和1. 当一个人把一份数据传给另一个人时,计算机传递的是其实是二进制数字,但这些数字需要被还原为原始信息. 这个工作当然是 ...

  4. 【springboot读取配置文件】@ConfigurationProperties、@PropertySource和@Value

    概念: @ConfigurationProperties : 是springboot的注解,用于把主配置文件中配置属性设置到对于的Bean属性上 @PropertySource :是spring的注解 ...

  5. 1020 Tree Traversals——PAT甲级真题

    1020 Tree Traversals Suppose that all the keys in a binary tree are distinct positive integers. Give ...

  6. where & having 关键字

    where和having都是做条件筛选的 where执行的时间比having要早 where后面不能出现组函数 having后面可以出现组函数 where语句要跟在from后面 ,where 不能单独 ...

  7. 探索 .NET Core 依赖注入的 IServiceProvider

    在上一篇文章中,我们学习了Microsoft.Extensions.DependencyInjection中的IServiceCollection,包括服务注册转换为ServiceDescriptor ...

  8. 栈的数组模拟(非STL)

    #include<bits/stdc++.h> using namespace std; struct zhan{ int s[10000]; int top=0; void zhanpo ...

  9. vue3中的通过proxy实现双向数据绑定的原理

    1.什么是Proxy?它的作用是? 据阮一峰文章介绍:Proxy可以理解成,在目标对象之前架设一层 "拦截",当外界对该对象访问的时候,都必须经过这层拦截,而Proxy就充当了这种 ...

  10. SpringMVC-03 RestFul和控制器

    SpringMVC-03 RestFul和控制器 控制器Controller 控制器复杂提供访问应用程序的行为,通常通过接口定义或注解定义两种方法实现. 控制器负责解析用户的请求并将其转换为一个模型. ...