这道题不难想到这样的dp。

dp[字符串si] = 以si为结尾的最大总权值。

dp[si] = max(dp[sj]) ,1.j < i,2.sj是si的子串。

对于第二个条件,是一个多模版串匹配的问题,可以用AC自动机。

预先O(m)把AC自动机建好,然后动态更新AC自动机上的dp值,

匹配的时候,指向字符的指针移动总共是O(m),

而每个单词,fail指针走寻找后缀却是O(m),即使改成后缀链接也是O(n)。too slow!

找到一个单词后,需要避免找后缀,动态维护这个单词的dp值。

一开始所有单词的dp都是0。

更新的时候,dp[si]需要更新所有dp[sj],其中si是sj的后缀。

如果父节点是子节点的后缀,把所有的单词(包括空后缀)连接起来将会得到以空字符串为根的后缀链接树。

这样就变成一个更新子树的问题,dfs把树形转成线性以后可以用线段树来维护。

询问单点最大值,区间更新O(logn)。

复杂度O(mlogn)

潜在的坑点:

1.Trie个结点可能对应多个单词,如果只更新了其中一个单词的线性区间RE...(map,前向链表,vector都可以搞

/*********************************************************
* ------------------ *
* author AbyssFish *
**********************************************************/
#include<cstdio>
#include<cstring>
#include<queue>
#include<algorithm>
using namespace std; const int LEN = 3e5+;
const int MAXN = 2e4+; int W[MAXN], S[MAXN];
int N;
char s[LEN]; int hd[LEN];
int nx[MAXN], to[MAXN], ec; void add_e(int u,int v)
{
to[ec] = v;
nx[ec] = hd[u];
hd[u] = ec++;
}
#define eachedge int i = hd[u]; ~i; i = nx[i]
inline void init_g(int n){ memset(hd,-,n<<); ec = ; }
int L[MAXN], R[MAXN], dfs_clk; //string's linear suffix link tree id
const int ST_SIZE = <<;
int dp[ST_SIZE]; #define para int o = 1,int l = 0,int r = dfs_clk
#define lo (o<<1)
#define ro (o<<1|1)
#define Tvar int md = (l+r)>>1;
#define lsn lo,l,md
#define rsn ro,md,r
#define insd ql<=l&&r<=qr void build(para)
{
dp[o] = ;
if(r-l>){
Tvar
build(lsn);
build(rsn);
}
} void update(int ql,int qr,int v,para)
{
if(insd){
dp[o] = max(dp[o],v);
}
else {
Tvar
if(ql < md) update(ql,qr,v,lsn);
if(qr > md) update(ql,qr,v,rsn);
}
} int query(int p,para)
{
int re = ;
while(r-l>){
Tvar
if(p<md){
o = lo; r = md;
}
else {
o = ro; l = md;
}
re = max(re,dp[o]);
}
return re;
} const int sigma_size = , MAXND = LEN;
struct AhoCorasick_automata
{
#define idx(x) (x-'a')
int ch[MAXND][sigma_size]; int f[MAXND];
int last[MAXND];
int cnt; int val[MAXND];
int nx_val[MAXN];
void add_v(int o,int x)
{
nx_val[x] = val[o];
val[o] = x;
} int newNode()
{
int i = ++cnt;
memset(ch[i],,sizeof(ch[i]));
val[i] = ;
return i;
} void init()
{
cnt = -; newNode();
} int add(char *s,int id)
{
int u = , i, c;
for(i = ; s[i]; i++){
c = idx(s[i]);
if(!ch[u][c]){
ch[u][c] = newNode();
}
u = ch[u][c];
}
add_v(u,id);
return i;
} queue<int> q;
void getFail()
{
int u, c, v, r;
//f[0] = 0; last[0] = 0;
for(c = ; c < sigma_size; c++){
u = ch[][c];
if(u){
q.push(u);
f[u] = ;
last[u] = ;
}
}
while(!q.empty()){
r = q.front(); q.pop();
for(c = ; c < sigma_size; c++){
u = ch[r][c];
if(u){
q.push(u);
v = f[u] = ch[f[r]][c];
last[u] = val[v] ? v : last[v];
}
else ch[r][c] = ch[f[r]][c];
}
}
} void dfs(int u)
{
int le = dfs_clk++;
for(eachedge){
dfs(to[i]);
}
int ri = dfs_clk;
for(int id = val[u]; id; id = nx_val[id]){
L[id] = le; R[id] = ri;
}
} void buildTree()
{
init_g(cnt+);
for(int u = ; u <= cnt; u++)if(val[u]){
add_e(last[u],u);
}
dfs_clk = ;
dfs();
} void work()
{
int i,j,c,u,id;
int ans = , mx;
build();
for(i = ; i <= N; i++){
u = ; mx = ;
for(j = S[i-]; j < S[i]; j++){
c = idx(s[j]);
u = ch[u][c];
if(val[u]){
id = val[u];
mx = max(mx, query(L[id]));
}
else if(last[u]){
id = val[last[u]];
mx = max(mx, query(L[id]));
}
}
if(W[i] > ){
ans = max(ans, mx += W[i]);
update(L[i],R[i],mx);
}
}
printf("%d\n",ans);
} }ac; void solve()
{
scanf("%d",&N);
ac.init();
for(int i = ; i <= N; i++){
scanf("%s%d",s+S[i-],W+i);
S[i] = ac.add(s+S[i-],i)+S[i-];
}
ac.getFail();
ac.buildTree();
ac.work();
} //#define LOCAL
int main()
{
#ifdef LOCAL
freopen("data.txt","r",stdin);
#endif
int T, kas = ; scanf("%d",&T);
while(++kas <= T){
printf("Case #%d: ",kas);
solve();
}
return ;
}

HDU 4117 GRE Words的更多相关文章

  1. hdu 4117 -- GRE Words (AC自动机+线段树)

    题目链接 problem Recently George is preparing for the Graduate Record Examinations (GRE for short). Obvi ...

  2. hdu 4117 GRE Words AC自动机DP

    题目:给出n个串,问最多能够选出多少个串,使得前面串是后面串的子串(按照输入顺序) 分析: 其实这题是这题SPOJ 7758. Growing Strings AC自动机DP的进阶版本,主题思想差不多 ...

  3. hdu 4117 GRE Words (ac自动机 线段树 dp)

    参考:http://blog.csdn.net/no__stop/article/details/12287843 此题利用了ac自动机fail树的性质,fail指针建立为树,表示父节点是孩子节点的后 ...

  4. [HDU 4787] GRE Words Revenge (AC自动机)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4787 题目大意: 给你若干个单词,查询一篇文章里出现的单词数.. 就是被我水过去的...暴力重建AC自 ...

  5. 综合(奇技淫巧):HDU 5118 GRE Words Once More!

    GRE Words Once More! Time Limit: 5000/5000 MS (Java/Others)    Memory Limit: 512000/512000 K (Java/O ...

  6. ●HDU 4787 GRE Words Revenge

    题链: http://acm.hdu.edu.cn/showproblem.php?pid=4787 题解: AC自动机(强制在线构造) 题目大意: 有两种操作, 一种为:+S,表示增加模式串S, 另 ...

  7. HDU 5118 GRE Words Once More!

    题目链接:HDU-5118 题意:给定一个有向无环图,每条边有一个权值.标定一些特定节点为“特殊节点”.从节点1出发到某“特殊节点”结束的路径,称为一个“GRE单词”.单词由路径上的权值组成.给定一组 ...

  8. HDU 4787 GRE Words Revenge

    Description Now Coach Pang is preparing for the Graduate Record Examinations as George did in 2011. ...

  9. [GodLove]Wine93 Tarining Round #4

    比赛链接: http://acm.hust.edu.cn/vjudge/contest/view.action?cid=44903#overview 题目来源: 2011 Asia ChengDu R ...

随机推荐

  1. display inline-block 间隔

    1.如果li横排用display:inline-block; 则li之间不能有间隔 必须连着一起,所以才一般用float:left; .today-wrap{ position: relative; ...

  2. 第三次 Scrum Meeting

    第三次 Scrum Meeting 写在前面 会议时间 会议时长 会议地点 2019/4/7 20:00 60min 新主楼G411 附Github仓库:WEDO 例会照片 工作情况总结(4.5-4. ...

  3. vue项目echarts画布删除历史数据重新渲染数据

    vue用到echarts时,根据select多选下拉框进行echarts折线图渲染.发现折现只能增加不能减少,后来根据echarts API文档发现 调用方式: chart.setOption(opt ...

  4. linux 运维基础之http协议详解

    引言 这尼玛博客还得自己在这里写,难受一匹本来排版好的...每次都这样嗨....本内容属于借鉴资源,侵权删! HTTP是一个属于应用层的面向对象的协议,由于其简捷.快速的方式,适用于分布式超媒体信息系 ...

  5. 关于jqgrid的一些使用

    1.jqgrid如何切换中英文 在做电力监控系统的时候,根据项目的需要涉及到中英文的切换,一直纠结了好久没有好的办法,虽然我知道可以手动更改引入的js文件就可以更改中英文,但是动态的一直没有办法更改, ...

  6. HZAU 18——Array C——————【贪心】

    18: Array C Time Limit: 1 Sec  Memory Limit: 128 MBSubmit: 586  Solved: 104[Submit][Status][Web Boar ...

  7. JS && || 陷阱 javascript 逻辑与、逻辑或 【转】

    通常来说逻辑运算a&&b和a||b分别是逻辑与运算和逻辑或运算,返回的是一个布尔值,要么为true,要么为false. 比如在PHP里面a&&b返回类型永远是布尔值,非 ...

  8. .net使用redis入门笔记

    1.学习blog:http://www.cnblogs.com/yangecnu/p/Introduct-Redis-in-DotNET.html 2.redis官网:http://redis.io/ ...

  9. Java 访问权限控制- protected 关键字

    protected 关键字的真正内涵 文章来源:http://blog.csdn.net/justloveyou_/article/details/61672133 很多介绍Java语言的书籍(包括& ...

  10. 又到圣诞节,让你的网页下起雪(js特效)

    又到圣诞节,让你的网页下起雪(js特效) 在4年多前,我写过一个特效,就是让你的网页下起雨,它的效果就是在你打开的网站,雨点下满你的屏幕,恩,大概效果如下图: 当然这个效果还有一些附带项,比如风速.风 ...