[Codeforces613E]Puzzle Lover

清疚 2021-12-17 10:01 286阅读 0赞

Problem

给你2*n的格子,每个格子有一个字母,从任意一点出发,不重复的经过上下左右,生成要求的字符串。问有几种不同的走法。

Solution

1232176-20171026082631254-1456300025.png
分三段,左U型、中间、右U型。
分别枚举左边和右边的长度,中间一段用Dp来解决。
Dp[i][j][k],i,j,k表示当前在(i,j)位置,枚举到第k个字符。

Notice

特殊情况下有重复。

Code

  1. #include<cmath>
  2. #include<cstdio>
  3. #include<cstring>
  4. #include<iostream>
  5. #include<algorithm>
  6. using namespace std;
  7. #define sqz main
  8. #define ll long long
  9. #define reg register int
  10. #define rep(i, a, b) for (reg i = a; i <= b; i++)
  11. #define per(i, a, b) for (reg i = a; i >= b; i--)
  12. #define travel(i, u) for (reg i = head[u]; i; i = edge[i].next)
  13. const int INF = 1e9, mod = 1e9 + 7, Ha = 826036489, N = 2000;
  14. const double eps = 1e-6, phi = acos(-1.0);
  15. ll modd(ll a, ll b) {if (a >= b || a < 0) a %= b; if (a < 0) a += b; return a;}
  16. ll read(){ ll x = 0; int zf = 1; char ch; while (ch != '-' && (ch < '0' || ch > '9')) ch = getchar();
  17. if (ch == '-') zf = -1, ch = getchar(); while (ch >= '0' && ch <= '9') x = x * 10 + ch - '0', ch = getchar(); return x * zf;}
  18. void write(ll y) { if (y < 0) putchar('-'), y = -y; if (y > 9) write(y / 10); putchar(y % 10 + '0');}
  19. char S[2][N + 5], st[N + 5];
  20. ll f[2][N + 5][N + 5], mi[N + 5];
  21. int n, m;
  22. void Calc(ll &X, ll Y)
  23. {
  24. X += Y;
  25. if (X >= mod) X -= mod;
  26. }
  27. struct Hash
  28. {
  29. ll hash[N + 5];
  30. void Make(int n, char *s)
  31. {
  32. hash[0] = 0;
  33. rep(i, 1, n) hash[i] = (hash[i - 1] * 31 + s[i] - 'a') % Ha;
  34. }
  35. ll Cut(int l, int r)
  36. {
  37. return (hash[r] - hash[l - 1] * mi[r - l + 1] % Ha + Ha) % Ha;
  38. }
  39. }pre[2], suf[2], Comp;
  40. ll Solve(int flag)
  41. {
  42. ll T = 0;
  43. memset(f, 0, sizeof(f));
  44. rep(j, 1, n)
  45. {
  46. f[0][j][0] = f[1][j][0] = 1;
  47. rep(i, 0, 1)
  48. {
  49. rep(k, 2, min(n - j + 1, m / 2))
  50. if (Comp.Cut(m - 2 * k + 1, m - k) == pre[i].Cut(j, j + k - 1) && Comp.Cut(m - k + 1, m) == suf[1 - i].Cut(n - (j + k - 1) + 1, n - j + 1))
  51. if (2 * k != m || flag) Calc(T, f[i][j][m - 2 * k]);
  52. rep(k, 2, min(j, m / 2))
  53. if (Comp.Cut(k + 1, 2 * k) == pre[i].Cut(j - k + 1, j) && Comp.Cut(1, k) == suf[1 - i].Cut(n - j + 1, n - (j - k + 1) + 1))
  54. if (2 * k != m || flag) Calc(f[i][j + 1][2 * k], 1);
  55. }
  56. rep(i, 0, 1)
  57. rep(k, 0, m - 1)
  58. if (S[i][j] == st[k + 1])
  59. {
  60. Calc(f[i][j + 1][k + 1], f[i][j][k]);
  61. if (k + 2 <= m && S[1 - i][j] == st[k + 2])
  62. Calc(f[1 - i][j + 1][k + 2], f[i][j][k]);
  63. }
  64. rep(i, 0, 1) Calc(T, f[i][j + 1][m]);
  65. }
  66. return T;
  67. }
  68. int sqz()
  69. {
  70. scanf("%s%s%s", S[0] + 1, S[1] + 1, st + 1);
  71. n = strlen(S[0] + 1), m = strlen(st + 1);
  72. mi[0] = 1;
  73. rep(i, 1, 2000) mi[i] = (mi[i - 1] * 31) % Ha;
  74. rep(i, 0, 1)
  75. {
  76. pre[i].Make(n, S[i]);
  77. reverse(S[i] + 1, S[i] + n + 1);
  78. suf[i].Make(n, S[i]);
  79. reverse(S[i] + 1, S[i] + n + 1);
  80. }
  81. Comp.Make(m, st);
  82. if (m == 1)
  83. {
  84. printf("%I64d\n", Solve(1) % mod);
  85. return 0;
  86. }
  87. ll ans = 0;
  88. Calc(ans, Solve(1));
  89. reverse(st + 1, st + m + 1);
  90. Comp.Make(m, st);
  91. Calc(ans, Solve(0));
  92. if (m == 2)
  93. rep(i, 1, n)
  94. {
  95. if (S[0][i] == st[1] && S[1][i] == st[2]) Calc(ans, mod - 1);
  96. if (S[1][i] == st[1] && S[0][i] == st[2]) Calc(ans, mod - 1);
  97. }
  98. printf("%I64d\n", ans);
  99. return 0;
  100. }

转载于:https://www.cnblogs.com/WizardCowboy/p/7735547.html

发表评论

表情:
评论列表 (有 0 条评论,286人围观)

还没有评论,来说两句吧...

相关阅读

    相关 不上班的 613

    ![format_png][] 人生充满了随机性,很多事情都不在我们的计划之中。 比如此刻的我,于 2020 年的情人节,坐在苏黎世联邦理工学院(ETH),回顾着从离职到现

    相关 613-Golang的例程

    Go不支持多线程,叫例程,例程是Go自身提供的,线程是系统提供的 但是我们依然管它叫多线程 CPU是计算机的本体 现代计算机是多核的CPU,意味着如果同时一次只做1件

    相关 [Codeforces613E]Puzzle Lover

    Problem 给你2\n的格子,每个格子有一个字母,从任意一点出发,不重复的经过上下左右,生成要求的字符串。问有几种不同的走法。 Solution ![1232