【AC自动机+DP】USACO2012 JAN GOLD_Video Game Combos

【题目大意】

给你个模式串(每个长度≤15,1≤N≤20),串中只含有三种字母。求一长度为K(1≤K≤1000)的字符串,使得匹配数最大(重复匹配计多次),输出最大值。

【解题思路】

W老师给的题,然而我不会做。呜呜呜谢谢丁爷爷教我做题,神犇丁爷爷%%%。下面都是丁爷爷的话,和我没有关系。然而丁爷爷没有博客(也许是我不造?( •? ω •? )y)现在正在码USACO给的标答…

[email protected]丁爷爷

代码是我自己的,因为是用指针写的会有点长,丁爷爷的代码只有60+。总之祝丁爷爷继续超神下去……

  1 #include<iostream>
  2 #include<cstdio>
  3 #include<cstring>
  4 #include<algorithm>
  5 #include<queue>
  6 #define lnum 3
  7 using namespace std;
  8 const int MAXK=1000;
  9 const int MAXN=20*15+1;
 10 int cnt=-1;
 11 struct ACauto
 12 {
 13     int id;
 14     ACauto* next[lnum];
 15     ACauto* fail;
 16     ACauto()
 17     {
 18         id=++cnt;
 19         for (int i=0;i<lnum;i++) next[i]=NULL;
 20         fail=NULL;
 21     }
 22 };
 23 ACauto* rt=new ACauto();
 24 int go[MAXN][lnum];//编号为i的节点的三个后继的编号,如果不存在则为0
 25 int combo[MAXN];//编号为i的节点及其后缀能够产生的最大匹配数
 26 int dp[MAXN][MAXK];//在编号为i的节点上再走j步能够达到的最大值
 27 int n,k;
 28
 29 void insert(ACauto* rt,char* str)
 30 {
 31     int len=strlen(str);
 32     ACauto* now=rt;
 33     for (int i=0;i<len;i++)
 34     {
 35         int index=str[i]-‘A‘;
 36         if (now->next[index]==NULL)
 37         {
 38             now->next[index]=new ACauto();
 39         }
 40         go[now->id][index]=now->next[index]->id;
 41         now=now->next[index];
 42     }
 43     combo[now->id]=1;
 44     //在不包含后缀的情况下,当前结尾可以产生一个匹配
 45 }
 46
 47 void buildfail(ACauto* rt)
 48 {
 49     queue<ACauto*> que;
 50     que.push(rt);
 51     while (!que.empty())
 52     {
 53         ACauto* head=que.front();que.pop();
 54         for (int i=0;i<lnum;i++)
 55         {
 56             if (head->next[i]==NULL) continue;
 57             if (head==rt)
 58                 head->next[i]->fail=rt;
 59             else
 60             {
 61                 ACauto* tmp=head->fail;
 62                 while (tmp!=NULL)
 63                 {
 64                     if (tmp->next[i]!=NULL)
 65                     {
 66                         head->next[i]->fail=tmp->next[i];
 67                         break;
 68                     }
 69                     else
 70                     tmp=tmp->fail;
 71                 }
 72                 if (tmp==NULL) head->next[i]->fail=rt;
 73             }
 74             combo[head->next[i]->id]+=combo[head->next[i]->fail->id];
 75             //当前节点及其字符串后缀的节点均可能为模式串,故每次都要沿着指针计算这一步能够产生的新的匹配数。由于计算时累加的,只需沿fail指针走一步即可。
 76             que.push(head->next[i]);
 77         }
 78     }
 79 }
 80
 81 void init()
 82 {
 83     memset(go,0,sizeof(go));
 84     memset(combo,0,sizeof(combo));
 85     scanf("%d%d",&n,&k);
 86     for (int i=0;i<n;i++)
 87     {
 88         char str[1000];
 89         scanf("%s",str);
 90         insert(rt,str);
 91     }
 92     buildfail(rt);
 93 }
 94
 95 void dp_process()
 96 {
 97     memset(dp,0,sizeof(dp));
 98     for (int i=0;i<=cnt;i++) dp[0][i]=combo[i];
 99     int cur=0;
100     for (int l=1;l<=k;l++)
101     {
102         cur^=1;
103         for (int i=0;i<=cnt;i++)
104         {
105             dp[cur][i]=0;
106             for (int j=0;j<lnum;j++)
107                 dp[cur][i]=max(dp[cur][i],combo[i]+dp[cur^1][go[i][j]]);
108         }
109     }
110     printf("%d",dp[cur][0]);
111 }
112
113 int main()
114 {
115     init();
116     dp_process();
117     return 0;
118 }
时间: 2024-10-07 10:26:18

【AC自动机+DP】USACO2012 JAN GOLD_Video Game Combos的相关文章

HDU3341 Lost&#39;s revenge(AC自动机+DP)

题目是给一个DNA重新排列使其包含最多的数论基因. 考虑到内存大概就只能这么表示状态: dp[i][A][C][G][T],表示包含各碱基个数为ACGT且当前后缀状态为自动机第i的结点的字符串最多的数论基因数 其中ACGT可以hash成一个整数(a*C*G*T+c*G*T+g*T+T),这样用二维数组就行了,而第二维最多也就11*11*11*11个. 接下来转移依然是我为人人型,我是丢进一个队列,用队列来更新状态的值. 这题果然挺卡常数的,只好手写队列,最后4500msAC,还是差点超时,代码也

poj 1625 Censored!(AC自动机+DP+高精度)

题目链接:poj 1625 Censored! 题目大意:给定N,M,K,然后给定一个N字符的字符集和,现在要用这些字符组成一个长度为M的字符串,要求不包 括K个子字符串. 解题思路:AC自动机+DP+高精度.这题恶心的要死,给定的不能匹配字符串里面有负数的字符情况,也算是涨姿势 了,对应每个字符固定偏移128单位. #include <cstdio> #include <cstring> #include <queue> #include <vector>

HDU 2296 Ring AC自动机 + DP

题意:给你n个模式串,每个模式串有一个得分,让你构造出一个长度为N之内且分数最高的文本串;输出字典序列最小的. 解题思路:  AC自动机 + DP , 不过要输出字典序列最小,多开一个 一个三维字符串来辅助二维DP(新思路) , DP[i][j] ,表示到i位置状态为j的最大得分. 解题代码: 1 // File Name: temp.cpp 2 // Author: darkdream 3 // Created Time: 2014年09月11日 星期四 15时18分4秒 4 5 #inclu

HDU2296——Ring(AC自动机+DP)

题意:输入N代表字符串长度,输入M代表喜欢的词语的个数,接下来是M个词语,然后是M个词语每个的价值.求字符串的最大价值.每个单词的价值就是单价*出现次数.单词可以重叠.如果不止一个答案,选择字典序最小的. 题解:AC自动机+dp.dp[i][j]表示在字符串长度i,在自动机的第j个状态.因为要字典序最小,所以转移时要保持字典序最小. 想了各种转移姿势 最后还是查了题解 发现可以直接记录前缀转移…… #include <bits/stdc++.h> using namespace std; co

hdu 2296 aC自动机+dp(得到价值最大的字符串)

Ring Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submission(s): 3180    Accepted Submission(s): 1033 Problem Description For the hope of a forever love, Steven is planning to send a ring to Jane with a rom

hdu 2457 AC自动机+dp

DNA repair Time Limit: 5000/2000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submission(s): 2004    Accepted Submission(s): 1085 Problem Description Biologists finally invent techniques of repairing DNA that contains segments c

POJ1625 Censored!(AC自动机+DP)

题目问长度m不包含一些不文明单词的字符串有多少个. 依然是水水的AC自动机+DP..做完后发现居然和POJ2778是一道题,回过头来看都水水的... dp[i][j]表示长度i(在自动机转移i步)且后缀状态为自动机第j个结点的合法字符串数 dp[0][0]=1 转移转移... 注意要用高精度,因为答案最多5050. 还有就是要用unsigned char,题目的输入居然有拓展的ASCII码,编码128-255. 1 #include<cstdio> 2 #include<cstring&

HDU2296 Ring(AC自动机+DP)

题目是给几个带有价值的单词.而一个字符串的价值是 各单词在它里面出现次数*单词价值 的和,问长度不超过n的最大价值的字符串是什么? 依然是入门的AC自动机+DP题..不一样的是这题要输出具体方案,加个字符数组记录每个状态最优情况的字符串即可. 另外题目字典序是先考虑长度再考虑每一位单词:特别要注意,有一个非常坑的地方看了Disscus才知道——单词A包含单词B,那么只计算单词A不计算单词B. dp[i][j]表示长度i(自动机上转移k步)后缀状态是自动机第j个结点的字符串的最大价值 dp[0][

HDU2457 DNA repair(AC自动机+DP)

题目一串DNA最少需要修改几个基因使其不包含一些致病DNA片段. 这道题应该是AC自动机+DP的入门题了,有POJ2778基础不难写出来. dp[i][j]表示原DNA前i位(在AC自动机上转移i步)且后缀状态为AC自动机结点j的最少需要修改的基因数 转移我为人人型,从dp[i][j]向ATCG四个方向转移到dp[i+1][j'],如果结点被标记包含致病基因就不能转移. 1 #include<cstdio> 2 #include<cstring> 3 #include<que