回文串[APIO2014](回文树)
2021-03-20 14:24
阅读:349
4
首先建出s的回文树,然后对于每一个回文子串,记录cnt为它出现的次数。
对于它fail树上的儿子,肯定都是它的子串(后缀),所以他出现了cnt次,它的后缀也会出现cnt次。
我们从更新的节点从后往前遍历,假设现在遍历到i,就使cnt[fail[i]]+=cnt[i]。然后更新ans=max(ans, cnt[i]*len[i])。
#include#include #include using namespace std; const int N = 400000; char s[N]; int lens; long long ans; namespace Plalindromic_Tree{ struct node{ int go[26]; int fail, len; }pt[N]; int lst = 0, tot = 0; int cnt[N]; void build() { s[0] = -1; pt[++tot].len = -1; pt[0].fail = pt[1].fail = 1; } void add(int c, int n) { int p = lst; while (s[n - pt[p].len - 1] != s[n]) p = pt[p].fail; if (!pt[p].go[c]) { int v = ++tot, k = pt[p].fail; pt[v].len = pt[p].len + 2; while (s[n - pt[k].len - 1] != s[n]) k = pt[k].fail; pt[v].fail = pt[k].go[c]; pt[p].go[c] = v; } lst = pt[p].go[c]; cnt[pt[p].go[c]]++; } }using namespace Plalindromic_Tree; int main() { scanf("%s", s + 1); lens = strlen(s + 1); build(); for (int i = 1; i ) { add(s[i] - ‘a‘, i); } for (int i = tot; i; i--) { cnt[pt[i].fail] += cnt[i]; ans = max(ans, 1ll * cnt[i] * pt[i].len); } cout ans; return 0; }
评论
亲,登录后才可以留言!