题目链接: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. 【POJ】【2699】The Maximum Number of Strong Kings

    网络流/最大流/二分or贪心 题目大意:有n个队伍,两两之间有一场比赛,胜者得分+1,负者得分+0,问最多有几只队伍打败了所有得分比他高的队伍? 可以想到如果存在这样的“strong king”那么一 ...

  2. 序列化Color对象

    如下: public class Class2 { [XmlIgnore] public Color Color1 { get { return color1; } set { color1 = va ...

  3. Detecting diabetic retinopathy in eye images

    Detecting diabetic retinopathy in eye images The past almost four months I have been competing in a  ...

  4. 使用css3伪元素制作时间轴并且实现鼠标选中高亮效果

    利用css3来制作时间轴的知识要点:伪元素,以及如何在伪元素上添加锚伪类 1)::before 在元素之前添加内容. 2)::after 在元素之后添加内容. 提示:亦可写成 :before :aft ...

  5. iOS7 状态栏 修改为白色字体的步骤

    1在Info.plist中设置UIViewControllerBasedStatusBarAppearance 为NO2 在需要改变状态栏颜色的ViewController中在ViewDidLoad方 ...

  6. jQuery中的Deferred-详解和使用

    首先,为什么要使用Deferred? 先来看一段AJAX的代码: var data; $.get('api/data', function(resp) { data = resp.data; }); ...

  7. DevExpress Form那些事儿

    1:设置子窗体依附父窗体 首先将父窗体的属性中  IsMdiContainer 设置为 True   , 就是将窗体设置为 MDI窗体.子窗体和父窗体都是继承自RibbonForm的. 代码如下 : ...

  8. Android调用天气预报的WebService简单例子

    下面例子改自网上例子:http://express.ruanko.com/ruanko-express_34/technologyexchange5.html 不过网上这个例子有些没有说明,有些情况不 ...

  9. odata

    http://www.odata.org/ Open Data Protocol (开放数据协议,OData)是用来查询和更新数据的一种Web协议,其提供了把存在于应用程序中的数据暴露出来的方式.OD ...

  10. hdu 1848 Fibonacci again and again (初写SG函数,详解)

    思路: SG函数的应用,可取的值为不连续的固定值,可用GetSG求出SG,然后三堆数异或. SG函数相关注释见代码: 相关详细说明请结合前一篇博客: #include<stdio.h> # ...