题目链接:http://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&page=show_problem&problem=1960

题意:给出一个n*m的字母矩阵T和一个x*y的字母矩阵S。求S在T中出现了多少次?

思路:将S的每行看做一个串插入ac自动机。用T的每一行去匹配。那么我们可以得到每一次匹配都匹配了S的哪些行的串以及在T的这一行的哪个位置匹配到这个S的串。我们用f[i][j]表示以(i,j)为左上角的T,匹配了S的多少行。设某一次T的第r行匹配了S的第i行,位置是[c,c+y-1],那么令f[r-i][c]++。最后f[i][j]=x的就是能够完整匹配S的位置。

struct node
{
    int c[26];
    int fail;
    int a[105],aNum;
    
    void init()
    {
        clr(c,0); fail=-1;
        aNum=0;
    }
};

node a[N];
int cnt;

void insert(char *s,int id)
{
    int i,x,p=0;
    for(i=0;s[i];i++)
    {
        x=s[i]-'a';
        if(!a[p].c[x]) 
        {
            a[p].c[x]=++cnt;
            a[cnt].init();
        }
        p=a[p].c[x];
    }
    x=++a[p].aNum;
    a[p].a[x]=id;
}

void build()
{
    queue<int> Q;
    int i,u,p,q;
    Q.push(0);
    while(!Q.empty())
    {
        u=Q.front();
        Q.pop();
        
        FOR0(i,26)
        {
            if(a[u].c[i]!=0)
            {
                p=a[u].c[i];
                q=a[u].fail;
                if(q!=-1) a[p].fail=a[q].c[i];
                else a[p].fail=0;
                Q.push(p);
            }
            else
            {
                q=a[u].fail;
                if(q!=-1) a[u].c[i]=a[q].c[i];
                else a[u].c[i]=0;
            }
        }
    }
}

int f[1005][1005];
int n,m,X,Y;
char T[1005][1005],S[105][105];

void match(char *s,int r)
{
    int p=0,i,x,c,j,k;
    for(i=0;s[i];i++)
    {
        x=s[i]-'a';
        p=a[p].c[x];
        if(a[p].aNum) 
        {
            c=i-Y+1;
            FOR1(k,a[p].aNum) if(r>=a[p].a[k])
            {
                f[r-a[p].a[k]][c]++;
            }
        }
        j=a[p].fail;
        while(j>0)
        {
            if(a[j].aNum)
            {
                c=i-Y+1;
                FOR1(k,a[j].aNum) if(r>=a[j].a[k])
                {
                    f[r-a[j].a[k]][c]++;
                }
            }
            j=a[j].fail;
        }
    }
}

int main()
{
    rush()
    {
        a[0].init(); cnt=0;
        int i,j;
        RD(n,m);
        FOR0(i,n) RD(T[i]);
        RD(X,Y);
        FOR0(i,X) RD(S[i]),insert(S[i],i);
        build(); clr(f,0);
        FOR0(i,n) match(T[i],i);
        int ans=0;
        FOR0(i,n) FOR0(j,m) if(f[i][j]>=X) ans++;
        PR(ans);
    }
}

UVA 11019 Matrix Matcher(ac自动机)的更多相关文章

  1. UVA 11019 Matrix Matcher 矩阵匹配器 AC自动机 二维文本串查找二维模式串

    链接:https://vjudge.net/problem/UVA-11019lrjP218 matrix matcher #include<bits/stdc++.h> using na ...

  2. UVA 11019 Matrix Matcher ( 二维字符串匹配, AC自动机 || 二维Hash )

    题目: 传送门 题意: 给你一个 n * m 的文本串 T, 再给你一个 r * c 的模式串 S: 问模式串 S 在文本串 T 中出现了多少次. 解: 法一: AC自动机 (正解) 670ms 把模 ...

  3. UVa 11019 Matrix Matcher - Hash

    题目传送门 快速的vjudge传送门 快速的UVa传送门 题目大意 给定两个矩阵S和T,问T在S中出现了多少次. 不会AC自动机做法. 考虑一维的字符串Hash怎么做. 对于一个长度为$l$的字符串$ ...

  4. uva 11019 Matrix Matcher

    题意:给出一个n*m的字符矩阵T,你的任务是找出给定的x*y的字符矩阵P在T中出现了多少次. 思路:要想整个矩阵匹配,至少各行都得匹配.所以先把P的每行看做一个模式串构造出AC自动机,然后在T中的各行 ...

  5. UVA - 11019 Matrix Matcher (二维字符串哈希)

    给你一个n*m的矩阵,和一个x*y的模式矩阵,求模式矩阵在原矩阵中的出现次数. 看上去是kmp在二维情况下的版本,但单纯的kmp已经无法做到了,所以考虑字符串哈希. 类比一维情况下的哈希算法,利用容斥 ...

  6. AC自动机(二维) UVA 11019 Matrix Matcher

    题目传送门 题意:训练指南P218 分析:一行一行的插入,一行一行的匹配,当匹配成功时将对应子矩阵的左上角位置cnt[r][c]++;然后统计 cnt[r][c] == x 的数量 #include ...

  7. UVA 11019 Matrix Matcher(哈希)

    题意 给定一个 \(n\times m\) 的矩阵,在给定一个 \(x\times y\) 的小矩阵,求小矩阵在大矩阵中出现的次数. \(1 \leq n,m \leq 1000\) \(1\leq ...

  8. UVA 11019 Matrix Matcher(二维hash + 尺取)题解

    题意:在n*m方格中找有几个x*y矩阵. 思路:二维hash,总体思路和一维差不太多,先把每行hash,变成一维的数组,再对这个一维数组hash变成二维hash.之前还在想怎么快速把一个矩阵的hash ...

  9. UVA - 11019 Matrix Matcher hash+KMP

    题目链接:传送门 题解: 枚举每一行,每一行当中连续的y个我们hash 出来 那么一行就是 m - y + 1个hash值,形成的一个新 矩阵 大小是 n*(m - y + 1), 我们要找到x*y这 ...

随机推荐

  1. OrzFAng系列–树 解题报告

    题目描述 方方方种下了三棵树,两年后,第二棵树长出了n个节点,其中1号节点是根节点. 给定一个n个点的树 支持两种操作 方方方进行m次操作,每个操作为: (1)给出两个数i,x,将第i个节点的子树中, ...

  2. Codeforces Round #216 (Div. 2)解题报告

    又范低级错误! 只做了两题!一道还被HACK了,囧! A:看了很久!应该是到语文题: 代码:#include<iostream> #include<];    ,m2=;    ;i ...

  3. 怎样把网站js文件合并成一个?几种方法可以实现

    我们在建网站时经常会用js特效代码以使页面更美观,比如js幻灯片代码.js下拉菜单等,但是网页特效一多,如果js文件没有合并的话会降低网站的性能,这时我们就要考虑合并js文件了,ytkah总结了以下几 ...

  4. 【算法】E.W.Dijkstra算术表达式求值

    算术表达式求值 我们要学习的一个栈的用例同时也是展示泛型的应用的一个经典例子,就是用来计算算术表达式的值,例如 ( 1 + ( ( 2 + 3 ) * ( 4 * 5 ) ) ) 如果将4乘以5,把3 ...

  5. Android Environment FAQ (Frequently Asked Question)

    1.how to find out the Eclipse Version From Eclipse Menu Help ----> About Eclipse It displayed as ...

  6. c++11 内存模型解读

    c++11 内存模型解读 关于乱序 说到内存模型,首先需要明确一个普遍存在,但却未必人人都注意到的事实:程序通常并不是总按着照源码中的顺序一一执行,此谓之乱序,乱序产生的原因可能有好几种: 编译器出于 ...

  7. css display visibility

    当visibility被设置为"hidden"的时候,元素虽然被隐藏了,但它仍然占据它原来所在的位置.注意,当元素被隐藏之后,就不能再接收到其它事件了. display属性就有一点 ...

  8. Android activity属性

    android:allowTaskReparenting 是否允许activity更换从属的任务,比如从短信息任务 切换到浏览器任务. android:alwaysRetainTaskState 是否 ...

  9. linux read命令详解

    read命令从键盘读取变量的值,通常用在shell脚本中与用户进行交互的场合.该命令可以一次读取多个变量的值,变量和输入的值都需要使用空格隔开. 语法 read(选项)(参数) 选项 -p:指定读取值 ...

  10. MongoDB (五) MongoDB 数据库操作

    一.MongoDB创建数据库: use 命令 MongoDB use DATABASE_NAME 用于创建数据库.该命令将创建一个新的数据库,如果它不存在,否则将返回现有的数据库. 语法: use D ...