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];
}