Leetcode 651题:4键键盘问题

Leetcode网站中的这道题是Plus会员专享题,为了加深记忆特此将思路及解法记录下来,便于后期复习(好吧,就是因为贫穷开不起会员)

1.题目描述

2.思路

这个算法基于这样⼀个事实,最优按键序列⼀定只有两种情况:
(1)要么⼀直按 A :A,A,...A(当 N ⽐较⼩时)。
(2)要么是这么⼀个形式:A,A,...C-A,C-C,C-V,C-V,...C-V(当 N ⽐较⼤时)。
因为字符数量少(N ⽐较⼩)时, C-A C-C C-V 这⼀套操作的代价相对⽐较⾼,可能不如⼀个个按 A ;⽽当 N ⽐较⼤时,后期 C-V 的收获肯定很⼤。这种情况下整个操作序列⼤致是:开头连按⼏个 A ,然后 C-A C-C组合再接若⼲ C-V ,然后再 C-A C-C 接着若⼲ C-V ,循环下去。换句话说,最后⼀次按键要么是 A 要么是 C-V 。

3.解答 

状态:剩余的敲击次数n

选择:有4种,分别是 A 、 C-A 、 C-C 、 C-V ( Ctrl 简写为 C )。

定义dp数组:dp[ i ] 表示经过 i 次敲击后,屏幕上最多显示 dp[ i ] 个A

base case:dp[ 0 ] = 0

状态转移:

for (int i = 1; i <= N; i++) {
        // 按 A 键
        dp[ i ] = dp[i - 1] + 1;

        //按 CV 键
        for (int j = 2; j < i; j++) {
                // 全选 & 复制 dp[j - 2],连续粘贴 i - j 次
                // 屏幕上共 dp[j - 2] * (i - j + 1) 个 A
                dp[ i ] = Math.max(dp[ i ], dp[j - 2] * (i - j + 1));
        }
}

完整代码: 

public int maxA(int N) {
    int[] dp = new int[N + 1];
    dp[0] = 0;
    for (int i = 1; i <= N; i++) {
        // 按 A 键
        dp[i] = dp[i - 1] + 1;
        // 按 CV 键
        for (int j = 2; j < i; j++) {
            // 全选 & 复制 dp[j-2],连续粘贴 i - j 次
            // 屏幕上共 dp[j - 2] * (i - j + 1) 个 A
            dp[i] = Math.max(dp[i], dp[j - 2] * (i - j + 1));
        }
    }
    // N 次按键之后最多有⼏个 A?
    return dp[N];
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值