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