390 消除游戏(约瑟夫问题)

1. 问题描述:

给定一个从1 到 n 排序的整数列表。首先,从左到右,从第一个数字开始,每隔一个数字进行删除,直到列表的末尾。第二步,在剩下的数字中,从右到左,从倒数第一个数字开始,每隔一个数字进行删除,直到列表开头。我们不断重复这两步,从左到右和从右到左交替进行,直到只剩下一个数字。返回长度为 n 的列表中,最后剩下的数字。

示例:

输入:
n = 9,
1 2 3 4 5 6 7 8 9
2 4 6 8
2 6
6

输出:
6

来源:力扣(LeetCode)
链接:https://leetcode-cn.com/problems/elimination-game

2. 思路分析:

这道题目属于约瑟夫环的相关问题,我们其实可以画一下图如下图所示,分奇数与偶数两种情况进行讨论,首先讨论偶数的情况,第1次有1,2,3...2k,删除的是奇数编号的数字,最终剩下2,4,6...2k,然后重新编号,因为下一次需要从右边开始删除所以我们可以从右边开始编号,为1,2,3...k,用f(2k)来表示当前有2k个数最后剩下的数字编号,编号为从左到右开始编号,因为我们是在一开始删除的时候是从数字多的一步步删除到只剩下一个数字,最后一个数字编号肯定是1,所以我们需要逆着推导回删除数字上一层对应的编号,也即需要将编号还原会上一层这么多个数字对应的编号,因为相邻两层一个是从右往下编号一个是从左往右编号,所以要想知道f(2k),那么就需要知道与f(k)编号之间的关系,我们可以知道实际上f(2k) = 1 + k - f(k),当前有k个数字,因为从左往右与从右往左是对称的所以编号肯定是1 + k - f(k),这样就可以将编号映射会上一层了。实际上可以发现奇数的情况也是一样的,在第一次的时候奇数会去掉最后那个数字所以就转化为了偶数的情况,所以两个情况对应的递推式子完全一样。(其实f(t)是对应着每一次是从右往左删除还是从左往右删除的编号)

3. 代码如下:

class Solution:
    def lastRemaining(self, n: int) -> int:
        if n == 1: return 1
        return 2 * (n // 2 + 1 - self.lastRemaining(n // 2))

 

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值