传送门:POJ - 3376

题意:给你n个字符串,两两结合,问有多少个是回文的;

题解:这个题真的恶心,我直接经历了5种错误类型 : ) ... 因为卡内存,所以又把字典树改成了指针版本的。

字符串s与字符串t组合是回文串的情况

1. len(s) > len(t), t的反串是 s 的前缀,且s剩下的部分是回文串 (比如s: abbcb  t: ba

2. len(s) = len(t), s = t 的反串(比如s: abc  t: cba

3. len(s) < len(t), s 是 t 的反串的前缀,且 t 的反串剩下的部分是回文串(比如 s: ba  t: bbcb

用拓展kmp求出每个字符串的最长回文前缀和后缀(分别用原串和反串进行exkmp,用反串和原串进行exkmp就可以求出了)然后用原串建trie,用反串去匹配。

1. 在trie中, 若串在非结点位置匹配完成, 则把该节点往下走有多少个回文串累加到答案。

2. 在trie中, 若在匹配串时遇上在这个结点结束的字符串, 那么看剩下的后缀是否是回文, 若是, 则把在该点结束的字符串数目累加到答案。

将所有的字符串都放在一个数组里,用一个数组记录每个串的起点,这样能节省空间。

如果内存超了不妨看看这个地方 int tree[26]; 是不是开了30,26就过了.....

  1 #include<iostream>
2 #include<algorithm>
3 #include<string.h>
4 #define ll long long
5 using namespace std;
6
7 ///next[i]: T[i]到T[m - 1]与T(模式串)的最长相同前缀长度;
8 ///extend[i]: S[i]到S[n - 1](原串)与T的最长相同前缀长度。
9
10 const int maxn=2000100;
11 int nt[maxn],ex[maxn],k=1,be[maxn],len[maxn];
12 bool tmp[maxn][2];
13 char a[maxn],b[maxn];
14
15 struct Node
16 {
17 int val;
18 int color;
19 int tree[26];
20 };
21 Node z[maxn];
22 int tot, root;
23
24 int newnode()
25 {
26 z[tot].val = 0;
27 z[tot].color = 0;
28 memset(z[tot].tree, -1, sizeof(z[tot].tree));
29 tot++;
30 return tot-1;
31 }
32
33 //预处理计算next数组
34 void GETNEXT(char *str,int len)
35 {
36 int i=0,j,po;
37 nt[0]=len; //初始化nt[0]
38 while(str[i]==str[i+1]&&i+1<len) i++; //计算nt[1]
39 nt[1]=i;
40 po=1; //初始化po的位置
41 for(i=2;i<len;i++){
42 if(nt[i-po]+i<nt[po]+po) nt[i]=nt[i-po]; //第一种情况,可以直接得到nt[i]的值
43 else{ //第二种情况,要继续匹配才能得到nt[i]的值
44 j=nt[po]+po-i;
45 if(j<0) j=0; //如果i>po+nt[po],则要从头开始匹配
46 while(i+j<len&&str[j]==str[j+i]) j++; //计算nt[i]
47 nt[i]=j;
48 po=i; //更新po的位置
49 }
50 }
51 }
52
53 //计算extend数组
54 void EXKMP(char *s1,int len,char *s2,int l2,int s,int flag) ///s1的后缀和s2的前缀匹配
55 {
56 int i=0,j,po;
57 GETNEXT(s2,len); //计算子串的next数组
58 while(s1[i]==s2[i]&&i<l2&&i<len) i++; //计算ex[0]
59 ex[0]=i;
60 po=0; //初始化po的位置
61 for(i=1;i<len;i++){
62 if(nt[i-po]+i<ex[po]+po) ex[i]=nt[i-po]; //第一种情况,直接可以得到ex[i]的值
63 else{ //第二种情况,要继续匹配才能得到ex[i]的值
64 j=ex[po]+po-i;
65 if(j<0) j=0; //如果i>ex[po]+po则要从头开始匹配
66 while(i+j<len&&j<l2&&s1[j+i]==s2[j]) j++; //计算ex[i]
67 ex[i]=j;
68 po=i; //更新po的位置
69 }
70 }
71 for(int i=0;i<l2;i++)
72 if(ex[i]+i==l2) tmp[i+s][flag]=1;
73 }
74
75 void insert(char *a,int len,int s)
76 {
77 int p=root;
78 for(int i=0;i<len;i++){
79 int c=a[i]-'a';
80 z[p].val+=tmp[i+s][0];
81 if(z[p].tree[c]==-1) {
82 z[p].tree[c]=newnode();
83 }
84 p=z[p].tree[c];
85 }
86 z[p].color++;
87 }
88
89 int query(char *a,int len,int s)
90 {
91 int p=root;
92 ll ans=0;
93 for(int i=0;i<len;i++){
94 int c=a[i]-'a';
95 p=z[p].tree[c];
96 if(p==-1) break;
97 if((i<len-1&&tmp[s+i+1][1])||i==len-1) ans+=z[p].color;
98 }
99 if(p!=-1) ans+=z[p].val;
100 return ans;
101 }
102
103 int main()
104 {
105 ios::sync_with_stdio(false);
106 cin.tie(0);
107 cout.tie(0);
108 int t;
109 cin>>t;
110 int l=0;
111 tot=0;
112 root=newnode();
113 for(int i=0;i<t;i++){
114 cin>>len[i]>>a+l;
115 be[i]=l;
116 l+=len[i];
117 for(int j=0;j<len[i];j++){
118 b[j+be[i]]=a[l-1-j];
119 }
120 EXKMP(a+be[i],len[i],b+be[i],len[i],be[i],0);
121 EXKMP(b+be[i],len[i],a+be[i],len[i],be[i],1);
122 insert(a+be[i],len[i],be[i]);
123 }
124 ll ans=0;
125 for(int i=0;i<t;i++){
126 ans+=query(b+be[i],len[i],be[i]);
127 }
128 cout<<ans<<endl;
129 return 0;
130 }

POJ - 3376 Finding Palindromes(拓展kmp+trie)的更多相关文章

  1. poj 3376 Finding Palindromes

    Finding Palindromes http://poj.org/problem?id=3376 Time Limit: 10000MS   Memory Limit: 262144K       ...

  2. POJ3376 Finding Palindromes —— 扩展KMP + Trie树

    题目链接:https://vjudge.net/problem/POJ-3376 Finding Palindromes Time Limit: 10000MS   Memory Limit: 262 ...

  3. POJ 3376 Finding Palindromes(扩展kmp+trie)

    题目链接:http://poj.org/problem?id=3376 题意:给你n个字符串m1.m2.m3...mn 求S = mimj(1=<i,j<=n)是回文串的数量 思路:我们考 ...

  4. POJ 3376 Finding Palindromes(manacher求前后缀回文串+trie)

    题目链接:http://poj.org/problem?id=3376 题目大意:给你n个字符串,这n个字符串可以两两组合形成n*n个字符串,求这些字符串中有几个是回文串. 解题思路:思路参考了这里: ...

  5. POJ 3376 Finding Palindromes (tire树+扩展kmp)

    很不错的一个题(注意string会超时) 题意:给你n串字符串,问你两两匹配形成n*n串字符串中有多少个回文串 题解:我们首先需要想到多串字符串存储需要trie树(关键),然后我们正序插入倒序匹配就可 ...

  6. POJ 3376 Finding Palindromes EX-KMP+字典树

    题意: 给你n个串串,每个串串可以选择和n个字符串拼接(可以自己和自己拼接),问有多少个拼接后的字符串是回文. 所有的串串长度不超过2e6: 题解: 这题由于是在POJ上,所以string也用不了,会 ...

  7. POJ - 3376 Finding Palindromes manacher+字典树

    题意 给n个字符串,两两拼接,问拼接后的\(n\times n\)个字符串中有多少个回文串. 分析 将所有正串插入字典树中,马拉车跑出所有串哪些前缀和后缀为回文串,记录位置,用反串去字典树中查询,两字 ...

  8. poj3376 Finding Palindromes【exKMP】【Trie】

    Finding Palindromes Time Limit: 10000MS   Memory Limit: 262144K Total Submissions:4710   Accepted: 8 ...

  9. Seek the Name, Seek the Fame POJ - 2752(拓展kmp || kmp)

    题意: 就是求前缀和后缀相同的那个子串的长度  然后从小到大输出 解析: emm...网上都用kmp...我..用拓展kmp做的  这就是拓展kmp板题嘛... 求出extend数组后  把exten ...

随机推荐

  1. Java入门-jdk安装与环境搭建

    计算机 计算机的组成:硬件+软件 1.硬件 1.1CPU 好比人的大脑 主要负责数据的运算以及控制 1.2内存 存储数据(临时存储) 缺点: 如果断电,数据丢失 如果程序关闭或退出,数据丢失 1.3硬 ...

  2. LeetCode394 字符串解码

    给定一个经过编码的字符串,返回它解码后的字符串. 编码规则为: k[encoded_string],表示其中方括号内部的 encoded_string 正好重复 k 次.注意 k 保证为正整数. 你可 ...

  3. 【RAC】打完补丁后,发现只有一台rac可以启动,另一台无法启动

    安装11Gr2单机asm后,打完11.2.0.3.7的psu后,发现启动不起来数据库,alert日志内容如下: Errors in file /u01/app/oracle/diag/rdbms/bd ...

  4. SSTI

    最牛bypass:https://blog.csdn.net/solitudi/article/details/107752717 SSTI的奇怪绕过姿势:https://blog.csdn.net/ ...

  5. fsutil比较有用的几个命令

    Fsutil:fsinfo 主要由专业支持者使用.列出所有驱动器,查询驱动器类型,查询卷信息,查询特定的 卷信息或文件系统统计信息. 语法参数 drives 列出计算机中所有的驱动器. drivety ...

  6. 如何用Python中自带的Pandas和NumPy库进行数据清洗

    一.概况 1.数据清洗到底是在清洗些什么? 通常来说,你所获取到的原始数据不能直接用来分析,因为它们会有各种各样的问题,如包含无效信息,列名不规范.格式不一致,存在重复值,缺失值,异常值等..... ...

  7. 24V降压5V芯片,5A,4.5V-30V输入,同步降压调节器

    PW2205开发了一种高效率的同步降压DC-DC转换器5A输出电流.PW2205在4.5V到30V的宽输入电压范围内工作集成主开关和同步开关,具有非常低的RDS(ON)以最小化传导损失.PW2205采 ...

  8. requests模块的基本使用

    requests模块的基本使用 基于网络请求的模块. 环境的安装:pip install requests 作用:模拟浏览器发起请求 分析requests的编码流程: 1.指定url 2.发起了请求 ...

  9. 前端面试之CSS常用的选择器!

    前端面试之CSS常用的选择器! 标签选择器 <style> /* <!-- 标签选择器 :写上标签名 -->*/ p { color: green; } div { color ...

  10. SQL Server 邮箱告警配置

    目录 配置数据库邮件 * 手动启用数据库邮件功能 * 配置数据库邮件 * 测试数据库邮件 实现 JOB 任务运行状态的检测 * 定义操作员 * 新建死锁警报 * 设置 SQL Server 代理 创建 ...