[关闭]
@ZCDHJ 2019-07-31T08:48:32.000000Z 字数 768 阅读 514

NOI2009 管道取珠

未分类


考虑 的实际意义,其实可以看成将原过程做两遍,得到相同结果的方案数。那么直接 DP 即可。

  1. #include <iostream>
  2. #include <cstdio>
  3. const int MAXN = 500;
  4. const int MOD = 1024523;
  5. int n, m;
  6. int dp[2][MAXN | 1][MAXN | 1];
  7. char a[MAXN | 1], b[MAXN | 1];
  8. int main() {
  9. scanf("%d %d", &n, &m);
  10. scanf("%s", a + 1);
  11. scanf("%s", b + 1);
  12. dp[1][0][1] = 1;
  13. dp[0][1][0] = 1;
  14. if (a[1] == b[1]) dp[1][0][0] = dp[0][1][1] = 1;
  15. for (int i = 0; i <= n; ++i) {
  16. for (int j = 0; j <= m; ++j) {
  17. for (int k = 0; k <= n; ++k) {
  18. int l = i + j - k;
  19. if (l < 0 || l > m || i + j <= 1) continue;
  20. int o = i & 1, oo = o ^ 1;
  21. dp[o][j][k] = 0;
  22. if (i > 0 && k > 0 && a[i] == a[k]) dp[o][j][k] = (dp[o][j][k] + dp[oo][j][k - 1]) % MOD;
  23. if (i > 0 && l > 0 && a[i] == b[l]) dp[o][j][k] = (dp[o][j][k] + dp[oo][j][k]) % MOD;
  24. if (j > 0 && k > 0 && b[j] == a[k]) dp[o][j][k] = (dp[o][j][k] + dp[o][j - 1][k - 1]) % MOD;
  25. if (j > 0 && l > 0 && b[j] == b[l]) dp[o][j][k] = (dp[o][j][k] + dp[o][j - 1][k]) % MOD;
  26. }
  27. }
  28. }
  29. printf("%d\n", dp[n & 1][m][n]);
  30. return 0;
  31. }
添加新批注
在作者公开此批注前,只有你和作者可见。
回复批注