제출 #249169

#제출 시각아이디문제언어결과실행 시간메모리
249169receed회문 (APIO14_palindrome)C++14
47 / 100
1092 ms2944 KiB
#include <iostream> #include <iomanip> #include <cstdio> #include <vector> #include <algorithm> #include <cmath> #include <cstring> #include <cassert> #include <string> #include <set> #include <map> #include <random> #include <bitset> #include <string> #include <unordered_set> #include <unordered_map> #include <deque> #include <queue> #define rep(i, n) for (int i = 0; i < (n); i++) #define all(x) (x).begin(), (x).end() #define rall(x) (x).rbegin(), (x).rend() using namespace std; using ll = long long; using ul = unsigned long long; using ld = long double; const int N = 600001, M = 1 << 19; string s; int mp[N], mx[N], ord[N], c[N], cnt[N], a[N], n, pc[N], pord[N], lcp[N], pos[N], tr[M * 2]; int getl(int p, int m, int v=1, int l=0, int r=M) { if (tr[v] >= m || p <= l) return -1; if (r - l == 1) return l; int x = getl(p, m, v * 2 + 1, (l + r) / 2, r); if (x != -1) return x; return getl(p, m, v * 2, l, (l + r) / 2); } int getr(int p, int m, int v=1, int l=0, int r=M) { if (tr[v] >= m || r <= p) return n; if (r - l == 1) return l; int x = getr(p, m, v * 2, l, (l + r) / 2); if (x != n) return x; return getr(p, m, v * 2 + 1, (l + r) / 2, r); } int sm(int x, int y) { return x + y >= n ? x + y - n : x + y; } int main() { ios_base::sync_with_stdio(0); cin.tie(0); cin >> s; n = s.size(); int l = 0, r = 0; rep(i, 2 * n - 1) { if (i <= 2 * r) mp[i] = min(mp[(l + r) * 2 - i], r * 2 - i + 1); if (i % 2 == 0) mp[i] = max(mp[i], 1); while (i - mp[i] - 1 >= 0 && (i + mp[i] + 1) / 2 < n && s[(i - mp[i] - 1) / 2] == s[(i + mp[i] + 1) / 2]) mp[i] += 2; if ((i + mp[i] - 1) / 2 > r) { l = (i + mp[i] + 1) / 2; r = (i + mp[i] - 1) / 2; } } int ps = 2 * n - 1; for (int i = n - 1; i >= 0; i--) { while (mp[ps] < ps - i * 2 + 1) ps--; mx[i] = ps - i * 2 + 1; } // rep(i, 2 * n - 1) // cout << mp[i] << '\n'; // rep(i, n) // cout << mx[i] << '\n'; n++; rep(i, n) { a[i] = (i == n - 1 ? 0 : s[i] - 'a' + 1); cnt[a[i] + 1]++; } rep(i, 27) cnt[i + 1] += cnt[i]; rep(i, n) ord[cnt[a[i]]++] = i; int cc = 1; rep(i, n - 1) { c[ord[i + 1]] = c[ord[i]]; if (s[ord[i]] != s[ord[i + 1]]) { c[ord[i + 1]]++; cc++; } } for (int l = 1; l < n; l *= 2) { rep(i, cc + 1) cnt[i] = 0; rep(i, n) { cnt[c[i] + 1]++; pord[i] = ord[i]; pc[i] = c[i]; } rep(i, cc) cnt[i + 1] += cnt[i]; rep(i, n) { int q = pord[i] - l; if (q < 0) q += n; ord[cnt[c[q]]++] = q; } c[ord[0]] = 0; rep(i, n - 1) { c[ord[i + 1]] = c[ord[i]]; if (pc[ord[i]] != pc[ord[i + 1]] || pc[sm(ord[i], l)] != pc[sm(ord[i + 1], l)]) c[ord[i + 1]]++; } cc = c[ord[n - 1]] + 1; } rep(i, n) pos[ord[i]] = i; cc = 0; rep(i, n) { cc = max(cc - 1, 0); // cc = 0; if (pos[i] == n - 1) cc = 0; else while (s[i + cc] == s[ord[pos[i] + 1] + cc]) cc++; lcp[pos[i]] = cc; } // rep(i, n - 1) // assert(s.substr(ord[i], lcp[i]) == s.substr(ord[i + 1], lcp[i])); // rep(i, n) // cout << lcp[pos[i]] << '\n'; rep(i, n) { // cout << ord[i] << '\n'; // cout << s.substr(ord[i]) << '\n'; } rep(i, n - 1) tr[M + i] = lcp[i]; for (int i = M - 1; i > 0; i--) tr[i] = min(tr[i * 2], tr[i * 2 + 1]); ll ans = 0; rep(i, n) { ll cl = getl(pos[i], mx[i]), cr = getr(pos[i], mx[i]); ans = max(ans, (cr - cl) * mx[i]); } cout << ans; }
#Verdict Execution timeMemoryGrader output
Fetching results...
#Verdict Execution timeMemoryGrader output
Fetching results...
#Verdict Execution timeMemoryGrader output
Fetching results...
#Verdict Execution timeMemoryGrader output
Fetching results...
#Verdict Execution timeMemoryGrader output
Fetching results...