理解题意之后,很自然的想到了用AC自动机搞,结果网上一搜,全是暴搜,按照自己的思想,AC自动机搞起,果然在提交了数次之后,看到了Accept。

AC自动机需要三个步骤:

第一步:建立字典树;

第二步:为字典树建立 Fail 指针 ;

第三步:进行匹配;

#include<iostream>
#include<string>
#include<string.h>
#include<cctype>
#include<queue>
#include<stdio.h> #define max(x,y) ((x) > (y) ? (x) : (y)) using namespace std ; struct Node {
Node *next[26] ;
Node *fail ;
bool is_over ;
int len ;
}; Node* new_node() {
Node *root = new Node ;
root ->fail = NULL ;
for(int i = 0 ; i< 26 ; i++)
root->next[i] = NULL ;
root ->is_over = false ;
return root ;
} void build_tree( Node *root , char *s ) {
int len = strlen(s) ;
for(int i = 0 ; i < len ; i++) {
if(root->next[s[i] - 'a'] == NULL)
root->next[s[i] - 'a'] = new_node() ;
root = root ->next[s[i] - 'a'] ;
}
root ->is_over = true ;
root ->len = len ;
} void build_fail(Node *root) {
Node *r = root ;
queue< Node* > q ;
q.push(root) ;
while(!q.empty()) {
r = q.front() ;
q.pop() ;
Node *p = r ;
for(int i = 0 ; i < 26 ; i++) {
if(p->next[i] != NULL) {
if(p == root) {
p->next[i]->fail = root ;
q.push(p->next[i]) ;
continue ;
}
while(r->fail->next[i] == NULL && r ->fail != root)
r = r->fail ;
if(r->fail ->next[i] != NULL)
p ->next[i]->fail = r ->fail ->next[i] ;
else
p ->next[i] ->fail = root ;
q.push(p->next[i]) ;
}
}
}
} int A_C(Node *root , char *s) {
int len = strlen(s) ;
Node *r = root ;
int count = 0 ;
for(int i = 0 ; i < len ; i++) {
if(!isalpha(s[i]))
continue ;
if(r->next[s[i]-'a'] != NULL) {
r = r ->next[s[i]-'a'] ;
if(r->is_over && !(islower(s[i+1])) && !(islower(s[i - r->len])) )
count++ ;
}
else {
while(r->next[s[i]-'a'] == NULL && r != root )
r = r ->fail ;
if(r ->next[s[i] - 'a'] != NULL)
r = r ->next[s[i] - 'a'] ;
else
r = root ;
if(r->is_over && (!(isalpha(s[i+1])) ) && !(islower(s[i - r->len])) )
count++ ;
}
} return count ;
} int main() {
int n , m ;
int t = 1 ;
while(scanf("%d%d",&n,&m) == 2) {
Node *root = new_node() ;
Node *r = root ;
while(n--) {
char s[30] ;
scanf("%s",&s) ;
getchar() ;
build_tree(root,s) ;
root = r ;
}
r->fail = NULL ;
root = r ;
build_fail(r) ;
char str[30][100] ;
char str1[30][100] ;
int count[30] = {0} ;
int ma = 0 ;
for(int i = 0 ; i < m ; i++) {
gets(str[i]) ;
strcpy(str1[i],str[i]) ;
int len = strlen(str[i]) ;
for(int j = 0 ; j < len ; j++)
if(isupper(str[i][j]))
str[i][j] += 32 ;
count[i] = A_C(root,str[i]) ;
ma = max(ma,count[i]) ;
}
cout << "Excuse Set #" << t++ << endl;
for(int k = 0 ; k < m ; k++)
if(count[k] == ma)
cout << str1[k] << endl ;
cout << endl ;
}
return 0 ;
}

AC自动机妙用的更多相关文章

  1. AC自动机讲解

    今天花了半天肝下AC自动机,总算啃下一块硬骨头,熬夜把博客赶出来.. 正如许多博客所说,AC自动机看似很难很妙,而事实上不难,但的确很妙.笼统地说,AC自动机=Trie+KMP,但是仅仅知道这个并没有 ...

  2. Aho-Corasick automaton(AC自动机)解析及其在算法竞赛中的典型应用举例

    摘要: 本文主要讲述了AC自动机的基本思想和实现原理,如何构造AC自动机,着重讲解AC自动机在算法竞赛中的一些典型应用. 什么是AC自动机? 如何构造一个AC自动机? AC自动机在算法竞赛中的典型应用 ...

  3. AC自动机——多个kmp匹配

    (并不能自动AC) 介绍: Aho-Corasick automaton,最经典的处理多个模式串的匹配问题. 是kmp和字典树的结合. 精髓与灵魂: ①利用trie处理多个模式串 ②引入fail指针. ...

  4. AC自动机板子题/AC自动机学习笔记!

    想知道484每个萌新oier在最初知道AC自动机的时候都会理解为自动AC稽什么的,,,反正我记得我当初刚知道这个东西的时候,我以为是什么神仙东西,,,(好趴虽然确实是个对菜菜灵巧比较难理解的神仙知识点 ...

  5. 洛谷 P3808 【模板】AC自动机(简单版) 题解

    原题链接 前置知识: 字典树.(会 \(\texttt{KMP}\) 就更好) 显然呢,本题用 字典树 和 \(\texttt{KMP}\) 无法解决问题. 所以我们发明了一个东西: \(\textt ...

  6. KMP,HASH,Trie,AC自动机

    我做个总结算了下午看了一下AC自动机和学习我的大生物(当然是多谢鑫神了)..完了要崩.. 1 KMP 只要是学过的人都觉得比较简单吧 但是学不会的人就感觉很难了,我是那种顿悟的然后感觉非常简单的人过程 ...

  7. 背单词(AC自动机+线段树+dp+dfs序)

    G. 背单词 内存限制:256 MiB 时间限制:1000 ms 标准输入输出 题目类型:传统 评测方式:文本比较   题目描述 给定一张包含N个单词的表,每个单词有个价值W.要求从中选出一个子序列使 ...

  8. 【AC自动机】背单词

    题意: 0 s v:添加价值为v的字符串s 1 t:查询t中含的s的权值和.(不停位置算多次) 思路: 在线AC自动机. 同学用过一个妙妙子的分块算法. 这里用二进制分组:通常用作把在线数据结构问题转 ...

  9. 基于trie树做一个ac自动机

    基于trie树做一个ac自动机 #!/usr/bin/python # -*- coding: utf-8 -*- class Node: def __init__(self): self.value ...

随机推荐

  1. 解决android开发webservice的发布与数据库连接的问题

    由于app后续开发的需要,移植了两次webservice和数据库,遇到了不少问题,也花费了很多时间,实践告诉我要学会寻找问题的根源,这样才能在开发中节省时间,尽快解决问题!好,废话不多说,转入正题…… ...

  2. 在VirtualBox虚拟机上采集Fedora15系统

    在VirtualBox虚拟机上采集Fedora15系统 第一部分:创建系统并磁盘分区 1.点击VirtualBox上的新建 2.添加名称,选择类型和版本,点下一步 3.写入内存(不要超过物理内存),点 ...

  3. lucene定义自己的分词器将其分成单个字符

    问题描写叙述:将一句话拆分成单个字符.而且去掉空格. package com.mylucene; import java.io.IOException; import java.io.Reader; ...

  4. 基于eclipse的mybatis映射代码自动生成的插件http://blog.csdn.net/fu9958/article/details/7521681

    基于eclipse的mybatis映射代码自动生成的插件 分类: JAVA 数据库 工具相关2012-04-29 00:15 2157人阅读 评论(9) 收藏 举报 eclipsegeneratori ...

  5. 揭密: M2和高房价到底谁推高了谁?

    近期,著名经济学家吴敬链向新华网等媒体表示:房价高的根本原本就是货币超发.近期十年来我国M2增长率(广义货币增长率)太快."钱太多了,有些人要保值就要投资买房,又因货币非常多购买力非常强.就 ...

  6. Android ROM 制作教程

    本文来自: 起点手机论坛 具体文章參考:http://www.qdppc.com/forum.php?mod=viewthread&tid=43751&fromuid=1 1.Andr ...

  7. iptables 配置

    #查看iptables现有规则 iptables -L -n #先允许所有,不然可能悲剧 iptables -P INPUT ACCEPT #清除所有默认规则 iptables -F #清除自定义规则 ...

  8. HTML5新增加的功能

    1.部分代码代替了以前的代码     例如: 获取焦点     旧:document.getElementById("price");.focus;     新:<input ...

  9. ADO知识的运用二(Day 28)

    可能是最近有些沉迷于电视剧了,所以,好多天都没写总结了.我知道这样不好,但又老管不住自己,也许是我自控力太差了吧......呵呵,如果在不写总结,可能以后就会成为习惯了.所以今天要赶紧补上...... ...

  10. SQL2008 'OFFSET' 附近有语法错误。 在 FETCH 语句中选项 NEXT 的用法无效。

    'OFFSET' 附近有语法错误.在 FETCH 语句中选项 NEXT 的用法无效. 说明: 执行当前 Web 请求期间,出现未经处理的异常.请检查堆栈跟踪信息,以了解有关该错误以及代码中导致错误的出 ...