【经典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 ...
随机推荐
- 深入理解C语言-深入理解void
void的字面意思是"无类型",void *则为"无类型指针",void *可以指向任何类型的数据 void含义 void几乎只有注释和限制程序的作用,定义一个 ...
- golang struct 转map 及 map[string]*Struct 初始化和遍历
package main import ( "encoding/json" "errors" "fmt" "reflect&quo ...
- 网络流三大算法【邻接矩阵+邻接表】POJ1273
网络流的基本概念跟算法原理我是在以下两篇博客里看懂的,写的非常好. http://www.cnblogs.com/ZJUT-jiangnan/p/3632525.html http://www.cnb ...
- poco编译与运行
1.引言 Poco C++库是: 一系列C++类库,类似Java类库,.Net框架,Apple的Cocoa; 侧重于互联网时代的网络应用程序 使用高效的,现代的标准ANSI/ISO C++,并基于ST ...
- python-文件常用操作
文件内建函数和方法: open()——打开文件 read()——输入 readline()——输入一行 seek()——文件内移动 write()——输出 close()——关闭文件 练习:
- quartus ii 粗略使用教程
重复刚刚做过的下载程序,不选sof文件,选择jic文件 选择program config然后点击start,观察开发板,断电在开启后仍然有效果,想要擦除开发板flash文件,可以点击取消program ...
- C语言基础练习——最大值及其位置(二维数组)
C语言基础练习——最大值及其位置(二维数组) 时间限制: 1 Sec 内存限制: 10 MB 题目描述 有一个n×m的矩阵,要求编程序求出: 每行元素的最大值,以及其所在的行号和列号.求出所有元素的 ...
- Python学习笔记:运算符
逻辑运算符: + 加 - 减 * 乘 / 除 % 取模-返回除法的余数 ** 幂-返回x的y次方 // 整除 比较运算符: == 等于-比较对象是否相等 != ...
- Neo4j Cypher语法(三)
目录 5 函数 5.1 谓词函数 5.2 标量函数 5.3 聚合函数 5.4 列表函数 5.5 数学函数 5.6 字符串函数 5.7 Udf与用户自定义函数 6 模式 6.1 索引 6.2 限制 7 ...
- Java 代码运行顺序
1.静态代码块,只执行一次,从上到下,先父类后子类 2.父类构造代码块,从上到下,然后父类构造方法,执行次数与实例化次数相关 3.子类构造代码块,从上到下,然后子类构造方法,执行次数与实例化次数相关