POJ 1204 Word Puzzles(AC自动机)
这题的数据卡在,如下:
5 5 3
ABCDE
FGHIJ
KLMNO
PQRST
UVWXY
PQR
RS
RST
puzzle中间的行中可以包含要查询的多个单词。这个问题很好解决,SearchDfa的时候别return就行了,一直搜到字符串的结尾。
但是这样做也有一个bug,如果数据是这样的:
4 7 5
ABCDEFG
HIJKLMN
OPQRSTU
VWXYZAK
RSTU
STU
TU
JKL
KL
也就是说查询单词里面,大单词包含着小单词,实际上是很难处理的。我的一个设想是遇到所有危险节点的时候,都访问,该节点的所有前缀节点,并记录该前缀的位置信息,也就是要搜索的单词。这个搜索的时候同样不能return,因为小单词后面可能有大单词。
不过也可以搜索所有的点,每个点的8个方向都搜索一下,不过时间开销太大。
对于这题的输入其实很好处理,每一个单词都设置一个唯一的标号,然后构造tire的时候,在每个叶子节点上标记上这个唯一个标号。这样就是唯一的表示,不需要再这个单词的每一个字符上都标志它的标号,因为其他单词来的时候还得改,没用。
代码
#include <iostream>
#include <stdio.h>
#include <string.h>
#include <vector>
#include <queue>
using namespace std;
const int maxn=1010;
const int LETTERS=26;
int nNodesCount=0;
int str_num=-1,l,c,w;;
struct CNode {
    CNode * pChilds[LETTERS];
    CNode * pPrev;
    bool bBadNode;
    int num;
    void Init() {
        memset(pChilds,0,sizeof(pChilds));
        pPrev=NULL;
        bool bBadNode=false;
    }
};
CNode Tree[maxn*maxn];
char puzzle[maxn][maxn];
char words[maxn][maxn];
int loc[maxn][3];
int d[8][2]={{-1,0},{1,0},{0,-1},{0,1},{-1,-1},{-1,1},{1,-1},{1,1}};
int pd[8]={0,4,6,2,7,1,5,3};
//上下左右 左上 右上 左下 右下 
void Insert(CNode * pRoot,char * s,int num)
{
    for (int i=0;s[i];i++) {
        if (pRoot->pChilds[s[i]-'A']==NULL) {
            pRoot->pChilds[s[i]-'A']=Tree+nNodesCount;
            nNodesCount++;
        }
        pRoot=pRoot->pChilds[s[i]-'A'];
    }
    pRoot->bBadNode=true;
    pRoot->num=num;
}
void BuildDfa()
{
    for (int i=0;i<LETTERS;i++) {
        Tree[0].pChilds[i]=Tree+1;
    }
    Tree[0].pPrev=NULL;
    Tree[1].pPrev=Tree;
    deque<CNode *> q;
    q.push_back(Tree+1);
    while (!q.empty()) {
        CNode * pRoot=q.front();
        q.pop_front();
        for (int i=0;i<LETTERS;i++) {
            CNode * p=pRoot->pChilds[i];
            if (p) {
                CNode * pPrev=pRoot->pPrev;
                while (pPrev) {
                    if (pPrev->pChilds[i]) {
                        p->pPrev=pPrev->pChilds[i];
                        if (p->pPrev->bBadNode) {
                            p->bBadNode=true;
                        }
                        break;
                    }
                    else {
                        pPrev=pPrev->pPrev;
                    }
                }
                q.push_back(p);
            }
        }
    }
}
void SearchDfa(int x,int y,int dir)
{
    CNode * p=Tree+1;
    while (x>=0&&x<l&&y>=0&&y<c) {
    	while (true) {
            if (p->pChilds[puzzle[x][y]-'A']) {
                p=p->pChilds[puzzle[x][y]-'A'];
                if (p->bBadNode) {
                	int len=strlen(words[p->num]);
                	loc[p->num][0]=x-(len-1)*d[dir][0];
                	loc[p->num][1]=y-(len-1)*d[dir][1];
                	loc[p->num][2]=pd[dir];
                }
                break;
            }
            else {
                p=p->pPrev;
            }
        }
        x+=d[dir][0];
        y+=d[dir][1];
	}
}
int main()
{
    nNodesCount=2;
    scanf("%d%d%d",&l,&c,&w);
    for (int i=0;i<l;i++) {
        scanf("%s",puzzle[i]);
    }
    for (int i=0;i<w;i++) {
        scanf("%s",words[i]);
        Insert(Tree+1,words[i],i);
    }
    BuildDfa();
	for (int i=0;i<l;i++) {
		for (int k=0;k<8;k++) {
			SearchDfa(i,0,k);
			SearchDfa(i,c-1,k);
		}
	}
	for (int j=0;j<c;j++) {
		for (int k=0;k<8;k++) {
			SearchDfa(0,j,k);
			SearchDfa(l-1,j,k);
		}
	}
    for (int i=0;i<w;i++) {
        printf("%d %d %c\n",loc[i][0],loc[i][1],loc[i][2]+'A');
    }
    return 0;
}
/*
5 5 3
ABCDK
EFGHN
IJKLS
MNOPD
QRSTP
4 7 5
ABCDEFG
HIJKLMN
OPQRSTU
VWXYZAK
RSTU
STU
TU
JKL
KL
*/
POJ 1204 Word Puzzles(AC自动机)的更多相关文章
- [poj] 1204 Word Puzzles || AC自动机
		原题 给个X*Y的字符串矩阵,W个询问,对每个询问输出这个串出现的位置及方向,共有8个方向,顺时针开始分别用A~H表示 AC自动机的板子题. 对于待匹配串建一个自动机,然后从矩阵的四周分别沿八个方向跑 ... 
- PKU 1204 Word Puzzles(AC自动机)
		题目大意:原题链接 给定一个字符串矩阵和待查找的单词,可以朝8个不同的方向查找,输出待查找单词第一个字母在矩阵中出现的位置和该单词被查到的方向. A~H代表8个不同的方向,A代表正北方向,其他依次以4 ... 
- POJ 1204 Word Puzzles | AC 自动鸡
		题目: 给一个字母矩阵和几个模式串,矩阵中的字符串可以有8个方向 输出每个模式串开头在矩阵中出现的坐标和这个串的方向 题解: 我们可以把模式串搞成AC自动机,然后枚举矩阵最外围一层的每个字母,向八个方 ... 
- 【 POJ - 1204 Word Puzzles】(Trie+爆搜|AC自动机)
		Word Puzzles Time Limit: 5000MS Memory Limit: 65536K Total Submissions: 10782 Accepted: 4076 Special ... 
- [POJ 1204]Word Puzzles(Trie树暴搜&AC自己主动机)
		Description Word puzzles are usually simple and very entertaining for all ages. They are so entertai ... 
- pku1204 Word Puzzles AC自动机 二维字符串矩阵8个方向找模式串的起点坐标以及方向 挺好的!
		/** 题目:pku1204 Word Puzzles 链接:http://poj.org/problem?id=1204 题意:给定一个L C(C <= 1000, L <= 1000) ... 
- poj 1204 Word Puzzles(字典树)
		题目链接:http://poj.org/problem?id=1204 思路分析:由于题目数据较弱,使用暴力搜索:对于所有查找的单词建立一棵字典树,在图中的每个坐标,往8个方向搜索查找即可: 需要注意 ... 
- POJ 题目1204 Word Puzzles(AC自己主动机,多个方向查询)
		Word Puzzles Time Limit: 5000MS Memory Limit: 65536K Total Submissions: 10244 Accepted: 3864 S ... 
- poj 2778 DNA Sequence ac自动机+矩阵快速幂
		链接:http://poj.org/problem?id=2778 题意:给定不超过10串,每串长度不超过10的灾难基因:问在之后给定的长度不超过2e9的基因长度中不包含灾难基因的基因有多少中? DN ... 
随机推荐
- 题解【洛谷P1841】[JSOI2007]重要的城市
			题面 题解 最短路图模板题. 介绍一下最短路图: 先对原图跑一边单源最短路,求出源点到每个点\(i\)的最短路\(dis[i]\). 接下来构建新图:对于一条边\((x,y,v)\),若\(dis[x ... 
- HTML5学习(2)语义化
			什么是语义化? 1.每一个HTML元素都有具体的含义,例: a元素:超链接,p元素:段落 2.所有的元素与展示效果无关 元素内容展示到页面中的效果,应该由CSS决定. 因为浏览器带有默认的CSS样式, ... 
- swagger2 常用注解的使用
			一.@Api 效果: @Api注解放在类上面,这里的value是没用的,tags表示该controller的介绍. 二 .@ApiOperation 效果: @ApiOperation注解用于放在方法 ... 
- thinkphp的模型操作
			先开个坑 WHERE篇 1, 模糊查询 where['keyword'] = [ 'like' , '%test%'] 2, 不等于,大于 ,小于 EQ 等于(=)NEQ 不等于(<& ... 
- Python爬虫连载7-cookie的保存与读取、SSL讲解
			一.cookie的保存与读取 1.cookie的保存-FileCookie.Jar from urllib import request,parse from http import cookieja ... 
- argument
			js中arguments的用法 了解arguments这个对象之前先来认识一下javascript的一些功能: 其实Javascript并没有重载函数的功能,但是Arguments对象能够模拟重载 ... 
- SpringMVC_001 第一个SpringMVC程序
			今天我们来学习第一个SpringMVC程序 一.配置开发方式 (1)首先建立一个SpringMVC web程序 (2)导入jar包 (3)建立UserController.java package ... 
- Android学习10
			SharedPreferences 今天练习了利用SharedPreferences保存登录用户名密码: layout布局: <?xml version="1.0" enco ... 
- Android App测试计划和设计测试矩阵
			Android APP :日程管理APP 测试计划(Test Plan): 编号 测试时间 测试类型 测试计划 1. 5.1~5.5 单元测试 单元测试是由程序员自己来完成,程序员有责任编写功能代码, ... 
- Codeforces Round #624 (Div. 3) F
			题意: 给出n的质点,带着初位置和速度: 如果中途两点可以相遇dis(i,j)=0: 如果不可以相遇,mindis(i,j): 求n个点的两两质点最小dis(i,j)之和 思路: 因为当初位置x和速度 ... 
