这道题不难想到这样的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. Hadoop Ecosytem

    There are a lot of Hadoop related projects which are open sourced and widely used by many componies. ...

  2. 023-将表单序列化为json对象

    使用jQuery将表单序列化为json对象,其中serializeJson方法的名字任意,serializeArray()这个jQuery提供的方法.this指的就是谁调用了这个方法. $.fn.se ...

  3. 06-struts2与ognl的结合

    1 参数接收 2 配置文件中 1 Demo2Action package www.test.c_config; import com.opensymphony.xwork2.ActionSupport ...

  4. nutz框架使用记录之Cnd.wrap

    这是对Cnd.wrap 官方用法 , 直接硬编码 , [JAVA]List<Person> crowd = dao.query(Person.class, Cnd.wrap("n ...

  5. Go语言下载网络图片或文件

    最近闲来无事, 于是就简单学习了下Go语言的基本的用法.由于实践才是最快的学习方法,所以这里就以下载网络图片或文件入手来学习Go语言 文件下载到本地,通常的思路就是先获得网络文件的 输入流 以及本地文 ...

  6. 深入学习keepalived之预备工作--线程

    1. 线程的定义 1.1 线程定义在scheduler.h文件中,其定义如下所示 /* Thread itself. */ typedef struct _thread { unsigned long ...

  7. Python爬虫《爬取get请求的页面数据》

    一.urllib库 urllib是Python自带的一个用于爬虫的库,其主要作用就是可以通过代码模拟浏览器发送请求.其常被用到的子模块在Python3中的为urllib.request和urllib. ...

  8. 从list中随机选出几个数,并按照原来的顺序排列

    需求: 从list中随机选出几个数,并按照原来的顺序排列(比如从list中随机选出6个数) 方案一: //若对象size大于6,则随机去除6个对象,并按照原来的顺序排列 while(list.size ...

  9. 使用jquery去掉时光轴头尾部的线条

    一.前言:以前做类似时光轴的结构,几乎都是一条灰色线飞流直下,没有尽头.今天这个线条是从第一个圆点到最后一个圆点,那么问题来了,内容的高度还不是固定的,线条的长度怎么确定?怎么就能刚刚好从第一个点到最 ...

  10. Web前端面试指导(九):盒子模型你是怎么理解的?

    问题分析 这道题问得比较宽泛,一定要找准切入点,如果切入点找不准,很容易乱答,甚至答偏,所以找准切入点是非常的重要的. 解答思路 1)盒子模型有两种,W3C和IE盒子模型 (1)W3C定义的盒子模型包 ...