Zuma

Zuma

题意

每次可以删掉连续的一段回文串,删掉后两边的串会拼起来,问最少多少次可以删完整个串。

思路

区间 dp,对于一个区间 \([l,r]\),如果 \(a[l - 1] = a[r + 1]\) 那么 \([l - 1, r + 1]\) 是可以通过区间 \([l, r]\) 来删掉 \([l - 1, r + 1]\),而所用的次数肯定是相同的,因为 \(a[l - 1]、a[r + 1]\) 是可以拼到 \([l, r]\) 的回文串的,所以一起删掉就好了。然后要统计由两个串拼起来的答案,可以枚举分界线,分两边处理,随后转区间 dp 即可。

code

点击查看代码
#include <iostream>

using namespace std;

const int MaxN = 510;

int dp[MaxN][MaxN], a[MaxN], n;

int main() {
  cin >> n;
  for (int i = 1; i <= n; i++) {
    cin >> a[i], dp[i][i] = 1;
  }
  fill(dp[1], dp[n + 1], 1);
  for (int len = 2; len <= n; len++) {
    for (int i = 1, j = i + len - 1; j <= n; i++, j++) {
      dp[i][j] = a[i] == a[j] ? dp[i + 1][j - 1] : 1e9;
      for (int k = i; k < j; k++) {
        dp[i][j] = min(dp[i][j], dp[i][k] + dp[k + 1][j]);
      }
    }
  }
  cout << dp[1][n] << endl;
  return 0;
}

posted @ 2023-08-11 11:27  yabnto  阅读(23)  评论(0编辑  收藏  举报