题:https://ac.nowcoder.com/acm/contest/4010/K
题意:用一些模式串凑成一个目标串,每个模式串有消耗,问组合的最小消耗,或不能组成输出-1;
分析:典型的AC自动机处理后在跳fail的过程中进行操作,这里操作就是dp计算最小。用dp[i]表示长串前ii位的最小代价,若有一个单词s是长串的前ii项的后缀,那么可以用dp[i−len(s)]+val(s)转移到dp[i]
#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const ll INF=1e18;
const int M=5e5+;
int trie[M][],fail[M],cnt;
ll val[M],dp[M],deep[M];
void insert(string s,ll v){
int root=,len=s.size();
for(int i=;i<len;i++){
if(!trie[root][s[i]-'a'])
trie[root][s[i]-'a']=++cnt;
root=trie[root][s[i]-'a'];
}
val[root]=min(val[root],v);
deep[root]=len;
}
void getfail(){
queue<int>que;
while(!que.empty())
que.pop();
for(int i=;i<;i++)
if(trie[][i]){
fail[trie[][i]]=;
que.push(trie[][i]);
}
while(!que.empty()){
int now=que.front();
que.pop();
for(int i=;i<;i++){
if(trie[now][i]){
fail[trie[now][i]]=trie[fail[now]][i];
que.push(trie[now][i]);
}
else
trie[now][i]=trie[fail[now]][i];
}
}
}
ll query(string s){
int now=,len=s.size();
for(int i=;i<len;i++){
now=trie[now][s[i]-'a'];
for(int j=now;j;j=fail[j]){
if(deep[j])
dp[i+]=min(dp[i+],dp[i+-deep[j]]+val[j]);
}
}
if(dp[len]>=INF)
return -;
return dp[len];
}
void init(){
for(int i=;i<M;i++)
dp[i]=val[i]=INF;
dp[]=val[]=;
}
int main(){
ios::sync_with_stdio(false);
cin.tie();
init();
int n;
string s;
cin>>n;
for(int i=;i<=n;i++){
ll v;
cin>>s>>v;
insert(s,v);
}
fail[]=;
getfail();
cin>>s;
cout<<query(s);
return ;
}