Ring HDU - 2296 AC自动机+简单DP和恶心的方案输出
题意:
就是现在给出m个串,每个串都有一个权值,现在你要找到一个长度不超过n的字符串,
其中之前的m个串每出现一次就算一次那个字符串的权值,
求能找到的最大权值的字符串,如果存在多个解,输出最短的字典序最小的串。
当最大全权值为0时输出空串。
输入最多100个子串,权值为不超过100的正整数。
每个子串长度至少为1,不超过10, n <= 50
如果不考虑方案输出,这题就变得相当简单了。
dp【i】【j】表示走到长度为 i 的时候 ,到AC自动机 j 这个节点所获得的最大权值和。
我一开始的做法是在dp的过程中获得那个最优方案。
最后死活过不去,就换了一种超级暴力的写法。
将所有情况保存下来,去一个个找字典序最小的方案。
#include <set>
#include <map>
#include <stack>
#include <queue>
#include <cmath>
#include <cstdio>
#include <string>
#include <vector>
#include <time.h>
#include <cstring>
#include <iostream>
#include <algorithm> #define pi acos(-1.0)
#define eps 1e-9
#define fi first
#define se second
#define rtl rt<<1
#define rtr rt<<1|1
#define bug printf("******\n")
#define mem(a, b) memset(a,b,sizeof(a))
#define name2str(x) #x
#define fuck(x) cout<<#x" = "<<x<<endl
#define sf(n) scanf("%d", &n)
#define sff(a, b) scanf("%d %d", &a, &b)
#define sfff(a, b, c) scanf("%d %d %d", &a, &b, &c)
#define sffff(a, b, c, d) scanf("%d %d %d %d", &a, &b, &c, &d)
#define pf printf
#define FIN freopen("../date.txt","r",stdin)
#define gcd(a, b) __gcd(a,b)
#define lowbit(x) x&-x
#define IO iOS::sync_with_stdio(false) using namespace std;
typedef long long LL;
typedef unsigned long long ULL;
const int maxn = 1e6 + ;
const int maxm = 8e6 + ;
const int INF = 0x3f3f3f3f;
const int mod = ; char str[][];
int n, m, cost[]; struct Aho_Corasick {
int next[][], fail[], End[];
int root, cnt; int newnode() {
for (int i = ; i < ; i++) next[cnt][i] = -;
End[cnt++] = ;
return cnt - ;
} void init() {
cnt = ;
root = newnode();
} void insert(char buf[], int id) {
int len = strlen(buf);
int now = root;
for (int i = ; i < len; i++) {
if (next[now][buf[i] - 'a'] == -) next[now][buf[i] - 'a'] = newnode();
now = next[now][buf[i] - 'a'];
}
End[now] += id;
} void build() {
queue<int> Q;
fail[root] = root;
for (int i = ; i < ; i++)
if (next[root][i] == -) next[root][i] = root;
else {
fail[next[root][i]] = root;
Q.push(next[root][i]);
}
while (!Q.empty()) {
int now = Q.front();
Q.pop();
End[now] += End[fail[now]];
for (int i = ; i < ; i++)
if (next[now][i] == -) next[now][i] = next[fail[now]][i];
else {
fail[next[now][i]] = next[fail[now]][i];
Q.push(next[now][i]);
}
}
} void debug() {
for (int i = ; i < cnt; i++) {
printf("id = %3d,fail = %3d,end = %3d,chi = [", i, fail[i], End[i]);
for (int j = ; j < ; j++) printf("%2d", next[i][j]);
printf("]\n");
}
}
} ac; int dp[][];
string path[][]; int main() {
// FIN;
int T;
sf(T);
while (T--) {
sff(n, m);
for (int i = ; i < m; ++i) scanf("%s", str[i]);
for (int i = ; i < m; i++) sf(cost[i]);
ac.init();
for (int i = ; i < m; ++i) ac.insert(str[i], cost[i]);
ac.build();
mem(dp, -);
for (int i = ; i <= n; i++)
for (int j = ; j < ac.cnt; j++)
path[i][j] = "";
int ans = , num1 = , num2 = ;
dp[][] = ;
for (int i = ; i < n; ++i) {
for (int j = ; j < ac.cnt; ++j) {
if (dp[i][j] == -) continue;
for (int k = ; k < ; ++k) {
int idx = ac.next[j][k];
if (dp[i + ][idx] < dp[i][j] + ac.End[idx]) {
dp[i + ][idx] = dp[i][j] + ac.End[idx];
path[i + ][idx] = path[i][j] + char(k + 'a');
} else if (dp[i + ][idx] == dp[i][j] + ac.End[idx] &&
path[i + ][idx] > path[i][j] + char(k + 'a')) {
//fuck(path[i][j]);
path[i + ][idx] = path[i][j] + char(k + 'a');
// fuck(path[i][j]); }
}
}
}
for (int i = ; i <= n; i++) {
for (int j = ; j < ac.cnt; ++j) {
if (ans < dp[i][j]) {
ans = dp[i][j], num1 = i, num2 = j;
} else if (ans == dp[i][j]&&path[num1][num2] > path[i][j] && path[num1][num2].size()>=path[i][j].size()) {
num1 = i, num2 = j;
}
}
}
// printf("%d\n", ans);
cout << path[num1][num2] << endl;
}
return ;
}
Ring HDU - 2296 AC自动机+简单DP和恶心的方案输出的更多相关文章
- 小明系列故事――女友的考验 HDU - 4511 AC自动机+简单DP
		
题意:自己看题目,中文体面. 题解: 把所有不能走的路径放入AC自动机中. 然后DP[i][j]表示走到 i 这个点,且位于AC自动机 j 这个节点最短距离 然后直接DP即可.注意一点会爆int #i ...
 - hdu 2296 aC自动机+dp(得到价值最大的字符串)
		
Ring Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total Submis ...
 - Walk Through Squares HDU - 4758 AC自动机+简单状压DP
		
题意:给你两个串,求用m个R,n个D能组成多少个包含这两个串 题解:先构造一个AC自动机记录每个状态包含两个串的状态, 状态很容易定义 dp[i][j][k][status]表示在AC自动机K这个节点 ...
 - Ring - HDU 2296(自动机+dp)
		
题目大意:斯蒂文想送给他女盆友一个戒指,并且他想在戒指上刻一些字,他非常了解他女盆友喜欢什么单词,比如"love""forvevr"....并且他还把女盆友喜欢 ...
 - 2016ACM/ICPC亚洲区沈阳站H - Guessing the Dice Roll HDU - 5955 ac自动机+概率dp+高斯消元
		
http://acm.hdu.edu.cn/showproblem.php?pid=5955 题意:给你长度为l的n组数,每个数1-6,每次扔色子,问你每个串第一次被匹配的概率是多少 题解:先建成ac ...
 - Censored! - POJ 1625(ac自动机+简单dp+高精度运算)
		
题目大意:首先给一个字符集合,这个集合有N个字符,然后需要一个长度为M的句子,但是据子里面不能包含的串有P个,每个串里面的字符都是有字符集和里面的字符构成的,现在想知道最多能构造多少个不重复的句子. ...
 - hdu 4057(ac自动机+状态压缩dp)
		
题意:容易理解... 分析:题目中给的模式串的个数最多为10个,于是想到用状态压缩dp来做,它的状态范围为1-2^9,所以最大为2^10-1,那我们可以用:dp[i][j][k]表示长度为i,在tri ...
 - hdu 2825 aC自动机+状压dp
		
Wireless Password Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others ...
 - hdu 2896 AC自动机
		
// hdu 2896 AC自动机 // // 题目大意: // // 给你n个短串,然后给你q串长字符串,要求每个长字符串中 // 是否出现短串,出现的短串各是什么 // // 解题思路: // / ...
 
随机推荐
- scala中Trait简单使用
			
trait Log { def log(message:String) = println("log:" + message) } /** * 为实例混入trait * */ tr ...
 - [原创] delphi KeyUp、KeyPress、Keydown区别和用法,如何不按键盘调用事件
			
KeyPress (Sender: TObject; var Key: Char); 当用户按下键盘上的字符键(字母,数字) 会触发该事件,功能键则不会(F1-F12,Ctrl,Alt,Shift ...
 - 通过jquery获取页面信息
			
获取浏览器显示区域(可视区域)的高度 : $(window).height(); 获取浏览器显示区域(可视区域)的宽度 :$(window).width(); 获取页面的文档高度 $(document ...
 - jar中没有主清单属性【解决办法】
			
<plugin> <groupId>org.apache.maven.plugins</groupId> <artifactId>maven-compi ...
 - sqlserver 调优(二)
			
良好的系统和数据库设计,优质的SQL编写,合适的数据表索引设计,甚至各种硬件因素:网络性能.服务器的性能.操作系统的性能,甚至网卡.交换机等.这篇文章主要讲到如何改善SQL语句,还将有另一篇讨论如何改 ...
 - DXP 常用功能
			
1. PCB布板篇 t+s: 从原理图对应到pcb图中 F11: 修改顶层,底层信息 v+f: 回到PCB板中 i+l: 放置到画的矩形框内 对齐功能: ctrl + shift + t: 顶部对齐 ...
 - prettier包升级后vue项目运行报错
			
今天用vue-cli新建vue项目的时候,发现项目怎么都跑不起来. 最后通过与以前项目作比较,发现prettier这个依赖的版本从原来的1.12.0升级成了1.13.1.我也不太清楚为什么升级后项目跑 ...
 - Day9 - 异步IO\数据库\队列\缓存
			
本节内容 Gevent协程 Select\Poll\Epoll异步IO与事件驱动 Python连接Mysql数据库操作 RabbitMQ队列 Redis\Memcached缓存 Paramiko SS ...
 - 【ARC073F】Many Moves
			
题目 一个显然的\(dp\),设\(dp_{i,j}\)表示其中一个棋子在\(x_i\)点,另一个棋子在\(j\)点的最小花费 显然\(dp_{i,j}\)有两种转移 第一种是把\(x_i\)上的棋子 ...
 - 树形结构_红黑树:平衡2X 哈夫曼树:最优2X
			
红黑树:平衡2X 哈夫曼树:最优2X 红黑树 :TreeSet.TreeMap 哈夫曼树 1. 将w1.w2.…,wn看成是有n 棵树的森林(每棵树仅有一个结点): 2. 在森林中选出根结点的权值最小 ...