题目链接:http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemId=3909

题意:给出两个矩阵A和B,找出最大的相同子矩阵S。输出S的高和宽以及在A和B中的位置。

思路:1、利用偏移量,其实就是枚举第二个矩阵跟第一个矩阵的那个位置开始对齐。再明白一点,就是,首先枚举B的(0,0)与A的哪个位置对齐;然后,枚举A的(0,0)与B的哪个位置对齐。这两个可以合并起来,偏移量范围为(-n2,-m2)-(n1,m1)。

2、然后就可以直接比较出对应位置的是不是相同。得到一个n2*m2的g矩阵,g[i][j]为0或者1。然后我们就是找这个g中最大的只包含1的子矩阵。我们一行一行处理,对于某一行i的某列j,我们用h[j]记录这列向上可匹配的长度,就是当前行的j列向上有连续的h[j]个1。然后,我们用l[j]和r[j]记录j列最左和最右匹配的长度。我们只需要从左向右扫描一遍就可求出l[i];然后从右向左扫描一遍就可求出r[j]数组。比如我们以计算l[j]为例。(1)对于当前的j,若j=0或者h[j-1]=0,那么left=j(left是一个临时变量);(2)若i=0(第一行)或者g[i-1][j]=0(上面一行为0),那么l[j]=left,因为这个l[j]是每行重复使用的,前面可能l[j]是记录的别的值,但是现在必须要置为当前的left,因为i=0和g[i-1][j]=0都等同于这是新的一行开始;(2)若left>l[j],l[j]=left。

3、从2的最后一句看出这个貌似是以h优先的。不妨设g的一种情况为

0 1 2 3 (列号)

0 0 1 1

1 1 1 1

1 1 1 1

显然最大是8.那么现在呢?我们转移完第一行后第二列的l为2,我们在2的最后一句是若left>l[j],l[j]=left。但是到达第二行是l[2]=0对吧,但是没有转移。此时求出的是6。但是这个程序并没有错。因为总有一个偏移量我们能够得到这个:

0 1 2 3 (列号)

1 1 1 1

1 1 1 1

此时就得到了8。所以,其实在一次中是以列优先的,也就是h。但是这并不会导致正确结果丢失,因为我们枚举的偏移量会覆盖每种情况。

int n1,m1,n2,m2;
char a[N][N],b[N][N];
int h[N],r[N],l[N],g[N][N];
int Max,X1,Y1,X2,Y2,rr,cc;

void deal(int ax,int ay)
{
    if(ax+n2<0||ay+m2<0) return; 
    clr(h,0);
    int i,j,k,left,right,n=n2,m=m2;
    FOR0(i,n) FOR0(j,m)
    {
        if(a[i+ax+50][j+ay+50]==b[i][j]) g[i][j]=1;
        else g[i][j]=0;
    }
    FOR0(i,n)
    {
        FOR0(j,m) g[i][j]?h[j]++:h[j]=0;
        FOR0(j,m) if(h[j])
        {
            if(!j||!h[j-1]) left=j;
            if(!i||!g[i-1][j]) l[j]=left;
            if(left>l[j]) l[j]=left;
        }
        FORL0(j,m-1) if(h[j])
        {
            if(j==m-1||!h[j+1]) right=j;
            if(!i||!g[i-1][j]) r[j]=right;
            if(right<r[j]) r[j]=right;
        }
        FOR0(j,m) if((r[j]-l[j]+1)*h[j]>Max)
        {
            Max=(r[j]-l[j]+1)*h[j];
            rr=h[j];
            cc=r[j]-l[j]+1;
            X1=i-h[j]+ax+1;
            Y1=l[j]+ay;
            X2=i-h[j]+1;
            Y2=l[j];
        }
    }
}

int main()
{
    Rush(n1)
    {
        RD(m1);
        int i,j;
        clr(a,0);
        for(i=50;i<n1+50;i++) RD(a[i]+50);
        RD(n2,m2);
        FOR0(i,n2) RD(b[i]);
        Max=0;
        for(i=-n2;i<n1;i++) for(j=-m2;j<m1;j++)
        {
            deal(i,j);
        }
        if(!Max) puts("0 0");
        else PR(rr,cc),PR(X1+1,Y1+1),PR(X2+1,Y2+1);
    }
    return 0;
}

ZOJ 3367 Counterfeit Money(最大相同子矩阵)的更多相关文章

  1. ZOJ 1074 最大子矩阵和

    Description Given a two-dimensional array of positive and negative integers, a sub-rectangle is any ...

  2. ZOJ 1074 To the Max(DP 最大子矩阵和)

    To the Max Time Limit: 2 Seconds      Memory Limit: 65536 KB Problem Given a two-dimensional array o ...

  3. 【转载】图论 500题——主要为hdu/poj/zoj

    转自——http://blog.csdn.net/qwe20060514/article/details/8112550 =============================以下是最小生成树+并 ...

  4. ZOJ 刷题记录 (。・ω・)ノ゙(Progress:31/50)

    [热烈庆祝ZOJ回归] P1002:简单的DFS #include <cstdio> #include <cstring> #include <algorithm> ...

  5. ZOJ 1859 Matrix Searching(二维线段树)

    http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemId=1859 Matrix Searching Time Limit: 10 Seco ...

  6. ACM 中 矩阵数据的预处理 && 求子矩阵元素和问题

            我们考虑一个$N\times M$的矩阵数据,若要对矩阵中的部分数据进行读取,比如求某个$a\times b$的子矩阵的元素和,通常我们可以想到$O(ab)$的遍历那个子矩阵,对它的各 ...

  7. ZOJ People Counting

    第十三届浙江省大学生程序设计竞赛 I 题, 一道模拟题. ZOJ  3944http://www.icpc.moe/onlinejudge/showProblem.do?problemCode=394 ...

  8. ZOJ 3686 A Simple Tree Problem

    A Simple Tree Problem Time Limit: 3 Seconds      Memory Limit: 65536 KB Given a rooted tree, each no ...

  9. [BZOJ1127][POI2008] KUP子矩阵

    Description 给一个n*n的地图,每个格子有一个价格,找一个矩形区域,使其价格总和位于[k,2k] Input 输入k n(n<2000)和一个n*n的地图 Output 输出矩形的左 ...

随机推荐

  1. 来自平时工作中的javascript知识的积累---持续补充中

    ① SeaJs和RequireJS最大的区别 解惑:来自豆友 ② javascript中如何判断undefined var exp = undefined; if (exp === undefined ...

  2. depthstencil buffer 不支持 msaa

    phyreengine dx11 MRT不支持 depth rendertarget 的msaa 他里面竟然只写着,// not supported yet !!!! 导致hdao 时开msaa的话, ...

  3. .NET设计模式(18):迭代器模式(Iterator Pattern)(转)

    概述 在面向对象的软件设计中,我们经常会遇到一类集合对象,这类集合对象的内部结构可能有着各种各样的实现,但是归结起来,无非有两点是需要我们去关心的:一是集合内部的数据存储结构,二是遍历集合内部的数据. ...

  4. in_array函数的第三个参数 strict

    看段代码 <?php $array = array('testing',0,'name'); var_dump($array); var_dump(in_array('foo', $array) ...

  5. Sqli-labs less 46

    Less-46 从本关开始,我们开始学习order by 相关注入的知识. 本关的sql语句为$sql = "SELECT * FROM users ORDER BY $id"; ...

  6. 在 OS X Yosemite 中部署Mesos

    1)从mesos的官网下载mesos的最新稳定版本:http://mesos.apache.org/downloads/,本文为mesos-0.22.1版本. 2)移动至你喜欢的目录(你在该目录下具有 ...

  7. DevSecOps简介(二)

    越来越多的组织机构开始采取 DevOps 实践,作为呼应,本文将概括强调很多人认为这一实践缺失的部分:安全.随着 NV (网络虚拟化) 和 NFV (网络功能虚拟化)的使用率逐步攀升,在开发和部署流程 ...

  8. 常见的NoSql系统使用场景分析--转载

    •Cassandra •特性:分布式与复制的权衡\根据列和键范围进行查询\BigTable类似的功能:列,列族\写比读快很多 •最佳适用:写操作较多,读比较少的时候.如果你的系统都是基于Java的时候 ...

  9. codeforces 430A Points and Segments (easy)(理解能力有待提高……)

    题目 //终于看懂题目了,,,, //一条线段里面不是每个坐标上都有要染色的点,所以为了满足条件,只能考虑那些给出坐标的点 //所以就要排序一下了,不能直接根据坐标0 1 0 1…… #include ...

  10. HDU 1217 Arbitrage (Floyd)

    Arbitrage http://acm.hdu.edu.cn/showproblem.php?pid=1217 Problem Description Arbitrage is the use of ...