390. Elimination Game

本文探讨了一种特殊的算法问题,即从有序整数列表中交替地从左至右和从右至左隔数删除,直至剩下最后一个数字。通过递归方法解决了这一问题,并给出了清晰的实现代码及时间复杂度分析。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

问题描述

There is a list of sorted integers from 1 to n. Starting from left to right, remove the first number and every other number afterward until you reach the end of the list.

Repeat the previous step again, but this time from right to left, remove the right most number and every other number from the remaining numbers.

We keep repeating the steps again, alternating left to right and right to left, until a single number remains.

Find the last number that remains starting with a list of length n.

Example:

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

Output:
6

题目链接:


思路分析

给一个数字,从左到右隔一个删一个,再从右往左隔一个删一个,直到只剩下最后一个元素,求最后剩下的这个元素

从左往右的情况,奇数偶数的情况都是一样的,都是只剩下偶数,所以就是2乘以从右往左的结果。再看从右往左的结果,如果是奇数,那么还是全是偶数的情况;如果是偶数,那么可以看作奇数的情况再减去1的结果。不断递归出结果即可。

代码
class Solution {
public:
    int lastRemaining(int n) {
        return forward(n);
    }

    int forward(int n){
        if(n == 1) return 1;
        if(n == 2) return 2;
            return 2*backward(n/2);
    }

    int backward(int n){
        if(n == 1 || n == 2) return 1;
        if(n & 1 == 1)
            return 2*forward(n/2);
        else
            return 2*forward(n/2) - 1;
    }
};

时间复杂度: O(logn) O ( l o g n )
空间复杂度: O(1) O ( 1 )


反思

评论区的鬼才一行做法,还不是很理解。

class Solution {
public:
    int lastRemaining(int n) {
        return n == 1 ? 1 : 2 * (1 + n / 2 - lastRemaining(n / 2));
    }
};
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值