题目链接: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. bzoj 1189 二分+最大流判定

    首先我们可以二分一个答案时间T,这样就将最优性问题 转化为了判定性问题.下面我们考虑对于已知的T的判定 对于矩阵中所有的空点bfs一次,得出来每个点到门的距离, 然后连接空点和每个能在t时间内到达的门 ...

  2. yebis 和phyreengine的集成

    被虐了几个礼拜阿, 暗无天日阿,花样被虐阿 设置 backbuffer commandbuffer这种问题还在其次,和他们的support要phyreengine 的sample就可以了 虐我千百遍的 ...

  3. std::vector<Channel2*> m_allChannels;容器,以及如何根据channelid的意义

    std::vector<Channel2*> m_allChannels;容器,以及如何根据channelid的意义 这个容器保存了所有客户端连接的channel Channel2* Li ...

  4. tcp 多线程与多进程调用close

    http://blog.csdn.net/russell_tao/article/details/13092727 大家知道,所谓线程其实就是“轻量级”的进程.创建进程只能是一个进程(父进程)创建另一 ...

  5. mysql修改数据库表权限

    ps:通常我用的是:1.“grant all on *.* to root@'%' identified by 'yourpassword';”——这个还可以顺带设置密码.2.“flush privi ...

  6. vi/vim使用指北 ---- Learning the vi and Vim Editors 读书 笔记

    vi/vim作为liux系统下最强大,最流行的文本编辑器之一.边看<Learning the vi and vim Editor>边学习vim,顺便做写简单的笔记,供以后查询. 没看这本书 ...

  7. jquery select处理

    JQuery 绑定select标签的onchange事件,弹出选择的值,并实现跳转.传参 js 处理 select :选中,删除,更改等 http://blog.csdn.net/wust_star/ ...

  8. POJ 1552

    #include<iostream> using namespace std; int main() { ]; int i,j; ; do{ sum=; ;num[i-]!=&&a ...

  9. ***PHP implode() 函数,将数组合并为字符串;explode() 函数,把字符串打散为数组

    实例 把数组元素组合为字符串: <?php $arr = array('Hello','World!','I','love','Shanghai!'); echo implode(" ...

  10. Struts2 直接返回字符串(可用于json)

    struts2可以在method中直接返回一个字符串而不是视图.让Action中的返回字符串的方法,直接return null;并在return之前用输出流输出字符串就可以了,跟servlet输出HT ...