左右交替删除奇偶位置数字直到剩下最后一个数字 Elimination Game

本文介绍了一种算法问题,即从1到n的有序整数列表中,通过交替从左到右和从右到左移除元素的方式找到最后一个剩余的数字。提供了两种不同的递归方法实现,帮助理解这一复杂但有趣的数学挑战。

问题:

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

解决:

【分析】

首先设输入数据为 n 时,最终结果为 f(n).很显然,有 f(1)=1,f(2)=2.
当 n=2k 时,从1,3,5,... ,2k-1 删除奇数项后,剩余的是2,4,6,... ,2k, 共 k 个偶数项。
当 n=2k+1 时,从1,3,5,... ,2k-1 ,2k+1 删除奇数项后,剩余的是2,4,6,... ,2k, 共 k 个偶数项。

这两种情况在从左到右删除一遍后,剩下的内容相同,所以最终剩下的一项也相同,所以我们可以得出结论:
f(2k+1)=f(2k) 所以,我们只需考虑 n 为偶数的情况。

 ① 我们用一个boolean型变量left2right,为true表示从左往右,为false表示从右往左遍历。

当n为1时,不论从左往右还是从右往左都返回1。

如果n大于1,且是从左往右的话,我们返回2倍的对n/2的从右往左的遍历;

如果是从右往左的话,稍稍麻烦一些,我们肯定还是要对n/2调用递归函数的,但是要分奇偶情况,如果n为奇数,返回2倍的对n/2的从左往右的遍历的值;如果n为偶数,2倍的对n/2的从左往右的遍历的值,再减去1。

递归方法解决:

class Solution { //80ms
    public int lastRemaining(int n) {
        return dfs(n,true);
    }
    public int dfs(int n,boolean left2right){
        if (n == 1) return 1;
        if (left2right){
            return 2 * dfs(n / 2,false);
        }else {
            return 2 * dfs(n / 2,true) - 1 + n % 2;
        }
    }
}

② 第一次从左往右删除的时候,奇数都被删掉了,剩下的都是偶数。如果我们对所有数都除以2,那么得到一个1到n/2的新数列。下一次我们从右往左删出,那么返回的结果应该是调用递归的结果lastRemaining(n / 2)在数组1到n/2之间的镜像。何为镜像,比如1, 2, 3, 4这个数字,2的镜像就是3, 1的镜像是4。

class Solution { //81ms
    public int lastRemaining(int n) {
        return n == 1 ? 1 : 2 * ( 1 + n / 2 - lastRemaining(n / 2));
    }
}

转载于:https://my.oschina.net/liyurong/blog/1596866

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值