【经典dp】hdu4622Reincarnation
呕 卡64M内存卡了好久
题目描述
题目大意
给出一个字符串 S,每次询问一个区间的本质不同的子串个数。$|S| \le 2000$.
题目分析
首先无脑$n^2$个set开起来:MLE
稍微想想这个东西是能够区间dp的:对于一个字符串$[l,r]$,它的存在首先给所有包含$[l,r]$的$f[i][j]$贡献$1$;考虑相同的字符串下一次出现的位置$[l',r']$,那么对所有包含$[l,r']$的$f[i][j]$贡献$-1$。最后再按照顺序累加一下即可。
第一次:std::map存一个哈希值上一次出现的位置,MLE
第二次:手写哈希表,$\mod 8000007$挂链,MLE
第三次:手写哈希表,$\mod 5000007$挂链,MLE
第四次:手写哈希表,$\mod 10007$挂链,TLE?
第五次:手写哈希表,$\mod 500007$挂链、查询时候顺带后续修改,MLE???
嗯?这啥玩意?
啧。后来意识到枚举不同长度的时候,哈希表里历史记录值都是没用的。所以每次枚举长度都应该重置哈希表。而且这样一来,哈希表模数和空间就可以开小了。
还听说这个经典问题有一个$O(n)$做法?
网上找了一下只有两个类似的是:
1.loj#6070. 「2017 山东一轮集训 Day4」基因 强制在线询问$s[l\cdots r]$中有多少本质不同的回文子串 $n\le 10^5,q\le 2\times10^5$
2.「湖南省队集训2018 Day2」有趣的字符串题 每个询问会询问一段区间的本质不同回文子串个数 $n\le 3\times 10^5,m\le 10^6$
不过这两个题都要用到回文子串的性质:“一个串的所有回文后缀可以被划分成不超过log个等差数列”。所以网上尚没有找到这个问题的$O(n)$解法?
#include<bits/stdc++.h>
typedef unsigned long long uint;
const int maxn = ;
const int maxp = ;
uint base = ; struct node
{
uint num;
int val;
node(uint a=, int b=):num(a),val(b) {}
}edges[maxp];
struct Hash_Table
{
#define MO 10007
int head[],nxt[maxp],edgeTot;
void init(){edgeTot=,memset(head, -, sizeof head);}
int query(uint x, int c)
{
for (int i=head[x%MO]; i!=-; i=nxt[i])
if (edges[i].num==x){
std::swap(c, edges[i].val);
return c;
}
edges[++edgeTot] = node(x, c), nxt[edgeTot] = head[x%MO], head[x%MO] = edgeTot;
return ;
}
#undef MO
}g;
int T,n,q,l,r;
char s[maxn];
int f[maxn][maxn];
uint pwr[maxn],hsh[maxn],val; uint hash(int l, int r)
{
return hsh[r]-hsh[l-]*pwr[r-l+];
}
int main()
{
// freopen("hdu4622.in","r",stdin);
// freopen("hdu4622.out","w",stdout);
pwr[] = ;
for (int i=; i<=; i++)
pwr[i] = pwr[i-]*base;
for (scanf("%d",&T); T; --T)
{
scanf("%s",s+);
n = strlen(s+);
for (int i=; i<=n; i++) hsh[i] = hsh[i-]*base+s[i]-'a'+;
for (int i=; i<=n; i++)
for (int j=i; j<=n; j++) f[i][j] = ;
for (int d=; d<=n; d++)
{
g.init();
for (int i=,pos; i+d-<=n; i++)
{
val = hash(i, i+d-), pos = g.query(val, i);
if (pos) --f[pos][i+d-];
++f[i][i+d-];
}
}
for (int i=n; i>=; i--)
for (int j=i; j<=n; j++)
f[i][j] += f[i+][j]+f[i][j-]-f[i+][j-];
for (scanf("%d",&q); q; --q)
scanf("%d%d",&l,&r), printf("%d\n",f[l][r]);
}
return ;
}
END
【经典dp】hdu4622Reincarnation的更多相关文章
- HDU 1003 Max Sum --- 经典DP
HDU 1003 相关链接 HDU 1231题解 题目大意:给定序列个数n及n个数,求该序列的最大连续子序列的和,要求输出最大连续子序列的和以及子序列的首位位置 解题思路:经典DP,可以定义 ...
- poj1458 求最长公共子序列 经典DP
Common Subsequence Time Limit: 1000MS Memory Limit: 10000K Total Submissions: 45763 Accepted: 18 ...
- NYOJ - 矩形嵌套(经典dp)
矩形嵌套时间限制:3000 ms | 内存限制:65535 KB 描述 有n个矩形,每个矩形可以用a,b来描述,表示长和宽.矩形X(a,b)可以嵌套在矩形Y(c,d)中当且仅当a<c,b< ...
- 51nod 1412 AVL树的种类(经典dp)
http://www.51nod.com/onlineJudge/questionCode.html#!problemId=1412 题意: 思路: 经典dp!!!可惜我想不到!! $dp[i][k] ...
- NYOJ 16 矩形嵌套(经典DP)
http://acm.nyist.net/JudgeOnline/problem.php?pid=16 矩形嵌套 时间限制:3000 ms | 内存限制:65535 KB 难度: ...
- poj 1050 To the Max 最大子矩阵和 经典dp
To the Max Description Given a two-dimensional array of positive and negative integers, a sub-rect ...
- CS Academy Distinct Neighbours(经典dp)
CS Academy Distinct Neighbours(经典dp) 题意: 求相邻无相同数字的合法的排列数 题解: 题解 先将相同的数字分为一类,假设共有n组 定义\(dp[i][j]\)表示前 ...
- 【经典dp 技巧】8.13序列
经典的拆绝对值 题目大意 给定$n$个具有顺序的序列,允许对每个序列循环移动.记第$i$个序列尾元素为$x$,$i+1$个序列首元素为$y$,定义其连接收益为$|x-y|*i$,求$n$个序列连接最大 ...
- POJ 1160:Post Office 邮局经典DP
Post Office Time Limit: 1000MS Memory Limit: 10000K Total Submissions: 17168 Accepted: 9270 Desc ...
随机推荐
- python中的FQA (python 学习篇 1)
Q:1. " if __name__=='__main__' " 这句是什么意思,可以不加吗? A: 如果单独运行该文件,则该模块的内容会被执行: 若运行的文件引用该 ...
- [转帖]中芯国际14nm秋季量产 7nm工艺或在2020年底问世
中芯国际14nm秋季量产 7nm工艺或在2020年底问世 https://news.mydrivers.com/1/641/641087.htm 中芯正在发力.. 今年秋天 14nm两场 明年底 7n ...
- PAT A1031 Hello World for U (20)
思路: 读取数组 int i = 0; while(cin >> word) { c[i] = word; i++; } 计算边长 int n1 = (length + 2) / 3; i ...
- PAT B1018.锤子剪刀布(20)
一个没有通过,不知道为何 #include <cstdio> int change(char c) { if(c == 'B') return 0; if(c == 'C') return ...
- PAT B1020 月饼(25)
题目描述 月饼是中国人在中秋佳节时吃的一种传统食品,不同地区有许多不同风味的月饼.现给定所有种类月饼的库存量.总售价.以及市场的最大需求量,请你计算可以获得的最大收益是多少. 注意:销售时允许取出一部 ...
- 如何使用加多宝(jdb)在linux下调试Java程序
毕业时写了一段时间的C,那时候调试使用gdb,后来转了java,当时就想java程序怎么调试,找了一下,果然,那就是jdk自带的jdb windows里是这样的 Linux下是这样的 一般我在linu ...
- 【计数dp】Array Without Local Maximums
参考博客:[CF1068D]Array Without Local Maximums(计数DP) [题意] n<=1e5 dp[i][j][k]表示当前第i个数字为j,第i-1个数字与第i个之间 ...
- 怎样理解String的slice(), subString(), substr()三个方法
String.prototype.slice() 是js字符串的切片工具方法, 用于对字符串做'裁剪'操作, 不改变原字符串. 'helloworld'.slice(0,5); // 'hello'; ...
- 国内有哪些好的JAVA社区
转载自 https://www.zhihu.com/question/29836842#answer-13737722 并发编程网 - ifeve.com 强烈推荐 ImportNew - 专注Jav ...
- JAVA问题String literal is not properly closed by a double-quote
String literal is not properly closed by a double-quote 这个错误:string字串没有以双引号结束String DBURL = "jd ...