题目链接

\(|S|<=10^5\),时间还是很宽松的。

允许我们使用线性/\(N\log N\)/甚至\(N \sqrt N\)的算法。

设\(l[i]\)表示以\(a[i]\)结尾的最长回文串,\(r[i]\)表示以\(a[i]\)开头的最长的回文串,

那么答案很显然就是\(\max_{i=1}^{len-1}l[i]+r[i+1]\)

怎么求?

回顾一下我们的马拉车算法

  1. for(int i = 1; i < len; ++i){
  2. if(i < maxright)
  3. hw[i] = min(hw[(mid << 1) - i], hw[mid] + mid - i); //min左边的参数是这个点的对称点的hw值,右边的是保证这个部分在这个大回文串之内
  4. else hw[i] = 1;
  5. while(a[i + hw[i]] == a[i - hw[i]]) ++hw[i]; //拓展
  6. if(hw[i] + i > maxright){ //更新右端点
  7. maxright = hw[i] + i;
  8. mid = i;
  9. }
  10. }

我们在每个\(i\)处理出\(hw[i]\)后更新\(i\)~\(i+hw[i]-1\)的\(l\)值,每个位置只需要更新一次就好了,因为我们是从左到右遍历的,因此第一次更新的一定是最优值。所以我们只需要定义一个变量\(p\),表示已经更新到哪里了,然后每次\(for(p->i+hw[i]-1)\),更新\(l\)值,如果\(p\)已经超过\(i+hw[i]-1\),是不会更新的,保证每个位置只被更新一次,也就是保证了时间复杂度是线性的。\(r\)也同理,反过来跑一遍就好了。

  1. #include <cstdio>
  2. #include <cstring>
  3. #include <algorithm>
  4. using namespace std;
  5. const int MAXN = 100010;
  6. char b[MAXN], a[MAXN << 1];
  7. int hw[MAXN << 1], l[MAXN], r[MAXN], ans, n, p = 1;
  8. int main(){
  9. scanf("%s", b);
  10. a[0] = a[1] = '#';
  11. int len = strlen(b);
  12. for(int i = 0; i < len; ++i)
  13. a[(i << 1) + 2] = b[i], a[(i << 1) + 3] = '#';
  14. int maxright = 0, mid; len = (len << 1) + 3;
  15. for(int i = 1; i < len; ++i){
  16. if(i < maxright)
  17. hw[i] = min(hw[(mid << 1) - i], hw[mid] + mid - i);
  18. else hw[i] = 1;
  19. while(a[i + hw[i]] == a[i - hw[i]]) ++hw[i];
  20. if(hw[i] + i > maxright){
  21. maxright = hw[i] + i;
  22. mid = i;
  23. }
  24. for(; p < i + hw[i]; ++p) l[p] = (p - i) + 1 - (a[p] == '#');
  25. }p = len - 1;
  26. for(int i = len - 1; i; --i)
  27. for(; p > i - hw[i]; --p)
  28. r[p] = (i - p) + 1 - (a[p] == '#');
  29. for(int i = 1; i < len - 1; ++i)
  30. ans = max(ans, l[i] + r[i + 1]);
  31. printf("%d\n", ans);
  32. return 0;
  33. }

【洛谷 P4555】 [国家集训队]最长双回文串 (Manacher)的更多相关文章

  1. 洛谷P4555 [国家集训队]最长双回文串(manacher 线段树)

    题意 题目链接 Sol 我的做法比较naive..首先manacher预处理出以每个位置为中心的回文串的长度.然后枚举一个中间位置,现在要考虑的就是能覆盖到i - 1的回文串中 中心最靠左的,和能覆盖 ...

  2. 洛谷 P4555 [国家集训队]最长双回文串 解题报告

    P4555 [国家集训队]最长双回文串 题目描述 顺序和逆序读起来完全一样的串叫做回文串.比如acbca是回文串,而abc不是(abc的顺序为abc,逆序为cba,不相同). 输入长度为\(n\)的串 ...

  3. 洛谷 P4555 [国家集训队]最长双回文串(Manacher)

    题目链接:https://www.luogu.com.cn/problem/P4555 首先明白两个回文串,那么要使两个回文串成立,那么我们只能把$'#'$作为中间节点. 然后我们跑一边Manache ...

  4. 洛谷 P4555 [国家集训队]最长双回文串

    链接: P4555 题意: 在字符串 \(S\) 中找出两个相邻非空回文串,并使它们长度之和最大. 分析: 直接使用马拉车算法求出每个点扩展的回文串.如果枚举两个回文串显然会超时,我们考虑切割一个长串 ...

  5. P4555 [国家集训队]最长双回文串

    P4555 [国家集训队]最长双回文串 manacher 用manacher在处理时顺便把以某点开头/结尾的最长回文串的长度也处理掉. 然后枚举. #include<iostream> # ...

  6. 【洛谷】P4555 [国家集训队]最长双回文串

    P4555 [国家集训队]最长双回文串 题源:https://www.luogu.com.cn/problem/P4555 原理:Manacher 还真比KMP好理解 解决最长回文串问题 转化为长度为 ...

  7. Manacher || P4555 [国家集训队]最长双回文串 || BZOJ 2565: 最长双回文串

    题面:P4555 [国家集训队]最长双回文串 题解:就.就考察马拉车的理解 在原始马拉车的基础上多维护个P[i].Q[i]数组,分别表示以i结尾最长回文子串的长度和以i开头的最长回文子串的长度 然后就 ...

  8. BZOJ.2565.[国家集训队]最长双回文串(Manacher/回文树)

    BZOJ 洛谷 求给定串的最长双回文串. \(n\leq10^5\). Manacher: 记\(R_i\)表示以\(i\)位置为结尾的最长回文串长度,\(L_i\)表示以\(i\)开头的最长回文串长 ...

  9. [国家集训队]最长双回文串 manacher

    ---题面--- 题解: 首先有一个直观的想法,如果我们可以求出对于位置i的最长后缀回文串和最长前缀回文串,那么我们枚举分界点然后合并前缀和后缀不就可以得到答案了么? 所以我们的目标就是求出这两个数列 ...

  10. P4555 [国家集训队]最长双回文串(回文树)

    题目描述 顺序和逆序读起来完全一样的串叫做回文串.比如acbca是回文串,而abc不是(abc的顺序为abc,逆序为cba,不相同). 输入长度为 n 的串 S ,求 S 的最长双回文子串 T ,即可 ...

随机推荐

  1. mysql ON DUPLICATE KEY UPDATE、REPLACE INTO

    INSERT INTO ON DUPLICATE KEY UPDATE 与 REPLACE INTO,两个命令可以处理重复键值问题,在实际上它之间有什么区别呢?前提条件是这个表必须有一个唯一索引或主键 ...

  2. 纯原生仿ES6的Object.assign,实现深度合并对象

    源码: function isObj(x){ var type = typeof x; return x !== null && (type === 'object' || type ...

  3. Qt Creater 制作汽车仪表盘

    最近项目用到了模拟仪表,网上下载大神编写的按个仪表Meter没有成功 转战 QWt 编译后,在creater中仍然无法使用,只可以在代码中使用 百度说是我编译的版本不对 扔到 开始做自己的 这个用到了 ...

  4. Android之Activity小结

    Acitivity: 四种状态:活动状态.暂停状态.停止状态.销毁状态 四种加载模式:standard ,singleTop,singleTask,singleInstance: 七大方法:onCre ...

  5. wangEditor编辑器中解析html图文信息问题

    在JS中,有一种方法:innerHTML 属性设置或返回表格行的开始和结束标签之间的 HTML. 也就是说,我们可以利用这个属性,把字符串转换为html代码,这样就可以被解析了. 其次,我们是需要在页 ...

  6. kvm-1

    yum install libvirt* virt-* qemu-kvm* -y systemctl start libvirtd.service systemctl status libvirtd. ...

  7. HDU 4722 Good Numbers(位数DP)(2013 ACM/ICPC Asia Regional Online ―― Warmup2)

    Description If we sum up every digit of a number and the result can be exactly divided by 10, we say ...

  8. systemtap没找到函数变量

    为啥systemtap没找到函数 hon@station6:~/codebox/stap/net$ sudo stap -L 'kernel.function("sock_recvmsg_n ...

  9. can be found for element 'tx:annotation-driven'

    错误描述: ERROR [ContainerBackgroundProcessor[StandardEngine[Catalina]]] (ContextLoader.java:308) - Cont ...

  10. 开发一个delphi写的桌面图标管理代码

    参加工作了就很少有时间去玩delphi了,这个适合初学者看看,大神勿喷 工具 delhpi7.0 access数据库 原则win下有安装office就可用 当初不太熟悉sqlite所有没用这做数据库. ...