这道题不难想到这样的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. 微信小程序实现图片上传功能

    前端: 微信开发者工具 后端:.Net 服务器:阿里云 这里介绍微信小程序如何实现上传图片到自己的服务器上 前端代码 data: { productInfo: {} }, //添加Banner bin ...

  2. http请求报头和响应报头(1)

    1.web端不可避免的http缓存机制,要理解缓存机制,先来了解下http的请求报文和响应报文的内容 2.请求报文  2.1请求行    请求行三部分组成:请求方法.URL以及版本协议 请求的方法有G ...

  3. linux对于zombie的处理

    @(Linux基础)[僵尸进程处理] 今天在服务器上推送项目的时候,突然发现很卡.就用top查看了一下,果然此事不简单啊. top - 10:39:16 up 20 days, 23:11, 2 us ...

  4. Farey Sequence(欧拉函数板子题)

    题目链接:http://poj.org/problem?id=2478 Farey Sequence Time Limit: 1000MS   Memory Limit: 65536K Total S ...

  5. Windows 7安装PlayReady出现“任务被禁用”错误信息

    问题描述: Windows 7的Windows media center中安装PlayReady时出现:错误信息:任务被禁用.(异常来自 HRESULT:0x80041326) 解决办法: 先请确认是 ...

  6. java读取excel(只是读取)

      最近做项目需要读取excel,在网上找了几个,都需要下载各种jar,下载好之后还是不能用,而且还分(xls xlsx)这两种格式, 最后找到个这个,不需要下载jar包,格式通吃,不过只是简单的读取 ...

  7. 绘图神器-matplotlib入门

    这次,让我们使用一个非常有名且十分有趣的玩意儿来完成今天的任务,它就是jupyter. 一.安装jupyter matplotlib入门之前,先安装好jupyter.这里只提供最为方便快捷的安装方式: ...

  8. VS 连接数据库报错:在与 SQL Server 建立连接时出现与网络相关的或特定于实例的错误

    VS报错:在与 SQL Server 建立连接时出现与网络相关的或特定于实例的错误.未找到或无法访问服务器.请验证实例名称是否正确并且 SQL Server 已配置为允许远程连接. (provider ...

  9. 10、选择框:ion-select

    !重点 multiple="true" 控制 选择框是 多选还是单选.true为 多选类似 checkbox. /* ---html----*/ <ion-content p ...

  10. 在 UWP 应用中创建、使用、调试 App Service (应用服务)

    在 Windows 10 中微软为 UWP 引入了 App Service (即应用服务)这一新特性用以提供应用间交互功能.提供 App Service 的应用能够接收来自其它应用传入的参数进行处理后 ...