剑指offer 70. 圆圈中最后剩下的数字

简介: 剑指offer 70. 圆圈中最后剩下的数字

题目描述

0,1,…,n−1 这 n 个数字 (n>0) 排成一个圆圈,从数字 0 开始每次从这个圆圈里删除第 m 个数字。

求出这个圆圈里剩下的最后一个数字。

数据范围

1≤n,m≤4000

样例

输入:n=5 , m=3
输出:3

方法一:递推 O(n)



这道题通过找规律可以得到下面这个公式:


旧编号 = (新编号 + m) % n


这么一看可能会比较懵,光看公式会不太理解,我们直接上例子,在计算过程中理解这个公式,假设 n = 5 且 m = 3 ,来模拟一下计算的过程。


第一步: 从数字 0 开始数三个,即删除数字 2 。

ee7a487931a94cbeb1a52823f2a9e733.png


第二步: 从数字 3 开始数三个,即删除数字 0

第三步: 从数字 1 开始数三个,即删除数字 4



第四步: 从数字 1 开始数三个,即删除数字 1

第五步: 现在只剩下一个数字,直接返回数字 3 即可。

你可能还会比较疑惑,上面这些跟那个公式有什么关系吗


重点来了,可以看到我每个图中都给每个数字进行重新编号,而最后答案数字 3 的编号为 0 ,因为只有一个数字了。那么我们该如何从这个新的编号得到数字 3 最一开始的编号呢。

我们一开始只知道 nm ,并不知道哪个才是最后剩下的数字,但是一定可以知道的是该数字的新编号一定是 0 ,现在我们开始往回推!

第一步: 数字 3 的新编号为 0 ,那么其上一轮的编号就为 (0 + 3) % 2 = 1


093b1d41755b4fb4954079cdd3d0f972.png

第二步: 数字 3 的新编号为 1 ,那么其上一轮的编号就为 (1 + 3) % 3 = 1

第三步: 数字 3 的新编号为 1 ,那么其上一轮的编号就为 (1 + 3) % 4 = 0



ef08945f4db94d52a8a8d554e65b7b61.png


第四步: 数字 3 的新编号为 0 ,那么其上一轮的编号就为 (0 + 3) % 5 = 3

这时得到的编号 3 就是答案 3 最开始的位置,所以我们可以得到开头的那个公式:

旧编号 = (新编号 + m) % n


先去递归到最后一轮,从已知的编号 0 往回推,就可以得到最终的答案。

class Solution {
public:
    int lastRemaining(int n, int m) {
        if (n == 1)    return 0;
        return (lastRemaining(n - 1, m) + m) % n;
    }
};


欢迎大家在评论区交流~









目录
相关文章
|
6月前
|
机器学习/深度学习 C语言
LeetCode | 17.04.消失的数字和189.旋转数组
17.04.消失的数字 OJ链接 这里题目要求在时间复杂度上O(n)我们介绍三种方法,看看哪种方法适合这道题~~
【Leetcode -441.排列硬币 -448.找到所有数组中消失的数字】
【Leetcode -441.排列硬币 -448.找到所有数组中消失的数字】
40 0
|
3月前
【刷题记录】——消失的数字、旋转数组
【刷题记录】——消失的数字、旋转数组
|
6月前
|
C++
【一刷《剑指Offer》】面试题 14:调整数组顺序使奇数位于偶数前面
【一刷《剑指Offer》】面试题 14:调整数组顺序使奇数位于偶数前面
|
6月前
【一刷《剑指Offer》】面试题 20:顺时针打印矩阵
【一刷《剑指Offer》】面试题 20:顺时针打印矩阵
|
6月前
每日一题——圆圈中最后剩下的数字(约瑟夫环问题)
每日一题——圆圈中最后剩下的数字(约瑟夫环问题)
|
6月前
|
机器学习/深度学习 Windows
剑指 Offer 62:圆圈中最后剩下的数字
剑指 Offer 62:圆圈中最后剩下的数字
41 0
消失的数字,旋转数组(leetcode 一题多解)
消失的数字,旋转数组(leetcode 一题多解)
|
算法
【算法挨揍日记】day03——双指针算法_有效三角形的个数、和为s的两个数字
【算法挨揍日记】day03——双指针算法_有效三角形的个数、和为s的两个数字
52 0
Leecode 面试题62. 圆圈中最后剩下的数字
Leecode 面试题62. 圆圈中最后剩下的数字
65 0