POJ2185 Milking Grid 题解 KMP算法
题目链接:http://poj.org/problem?id=2185
题目大意:求一个二维的字符串矩阵的最小覆盖子矩阵,即这个最小覆盖子矩阵在二维空间上不断翻倍后能覆盖原始矩阵。
题目分析:next函数的应用。需要枚举每一行每一列的字符串所对应的的 \(nxt[]\) 值,然后通过分析计算出最小的宽和最小的高。
具体分析
参考链接:https://blog.csdn.net/u013686535/article/details/52197467
一看这题,容易想出一种很直观的做法:求出每一行的最小重复串长度,取所有行的最小重复串长度的lcm为宽;对列也同样操作求出高。这种想法虽然很直观,但是否正确呢?
事实上,这种算法并不是正确的。如下面的这个反例:
2 8
ABCDEFAB
AAAABAAA
对于这个例子:第一行为6,第二行为5,6与5的最小公倍数为30,大于8则取8为宽,但明显是错误的。
但由于poj的测试数据太弱,以致使用这种方法的程序也可以通过。
下面介绍一下正解的做法。
首先是确定宽度:我们分别求出每行所有可能的重复子串长度,例如对于aaaa就有1、2、3和4,然后取每行都有的重复子串长度中最小的作为宽。
例如,对于上面的例子,第一行的重复子串长度只可能是6或8(显然整个串为一个重复子串也是可以的),第二行则可能是5、6、7或8,那么取它们都有的6和8当中最小值6,就是最小覆盖子矩阵的宽。
然后分成两步:
- 考虑到每行的列数比较小, \(1 \le C \le 75\) ,可以直接暴力枚举重复子串的长度,每列逐一检查是否与当前枚举的重复子串相对应。因此求宽度这个部分的时间复杂度为 \(O(R \times C^2)\) 。
 - 找到C以后我们再来一个个枚举R就可以了,时间复杂度也是 \(O(R \times C^2)\) 。
 
不过我这里采用标记的方式,时间复杂度降到 \(O(R \times C)\) 。
实现代码如下:
#include <iostream>
#include <string>
#include <cstring>
#include <vector>
using namespace std;
const int maxn = 404000;
int m, nxt[maxn];
string t;
int N, M;
char ch[10010][80];
int cnt_c[80], cnt_r[10010];
void cal_next() {
    m = t.length();
    for (int i = 0, j = -1; i < m; i ++) {
        while (j != -1 && t[j+1] != t[i]) j = nxt[j];
        nxt[i] = (j+1 < i && t[j+1] == t[i]) ? ++j : -1;
    }
}
void solve() {
    // 首先清空用于记录的cnt_c和cnt_r数组
    memset(cnt_c, 0 ,sizeof(cnt_c));
    memset(cnt_r, 0, sizeof(cnt_r));
    // 遍历每一行
    for (int i = 0; i < N; i ++) {
        t = "";
        for (int j = 0; j < M; j ++) t += ch[i][j];
        cal_next();
        int j = m-1;
        while (nxt[j] != -1) {
            j = nxt[j];
            cnt_c[m-1-j] ++;
        }
    }
    // 遍历每一列
    for (int i = 0; i < M; i ++) {
        t = "";
        for (int j = 0; j < N; j ++) t += ch[j][i];
        cal_next();
        int j = m-1;
        while (nxt[j] != -1) {
            j = nxt[j];
            cnt_r[m-1-j] ++;
        }
    }
    // 找到最小的cnt_c[i]==N,没有则M
    int cc = 1, rr = 1;
    for (; cc < M; cc ++) if (cnt_c[cc] == N) break;
    // 找到最小的cnt_r[i]==M,没有则N
    for (; rr < N; rr ++) if (cnt_r[rr] == M) break;
    cout << rr * cc << endl;
}
int main() {
    while (cin >> N >> M) {
        for (int i = 0; i < N; i ++) cin >> ch[i];
        solve();
    }
    return 0;
}
作者:zifeiy
POJ2185 Milking Grid 题解 KMP算法的更多相关文章
- poj2185 Milking Grid【KMP】
		
Milking Grid Time Limit: 3000MS Memory Limit: 65536K Total Submissions: 10084 Accepted: 4371 Des ...
 - POJ2185 Milking Grid 【lcm】【KMP】
		
Description Every morning when they are milked, the Farmer John's cows form a rectangular grid that ...
 - POJ 2185 Milking Grid(KMP)
		
Milking Grid Time Limit: 3000MS Memory Limit: 65536K Total Submissions: 4738 Accepted: 1978 Desc ...
 - 【kmp算法】poj2185 Milking Grid
		
先对每行求出所有可能的循环节长度(不需要整除). 然后取在所有行中都出现了的,且最小的长度为宽. 然后将每一行看作字符,对所有行求next数组,将n-next[n](对这些行来说最小的循环节长度)作为 ...
 - POJ2185 Milking Grid KMP两次(二维KMP)较难
		
http://poj.org/problem?id=2185 大概算是我学KMP简单题以来最废脑子的KMP题目了 , 当然细节并不是那么多 , 还是码起来很舒服的 , 题目中描写的平铺是那种瓷砖一 ...
 - [USACO2003][poj2185]Milking Grid(kmp的next的应用)
		
题目:http://poj.org/problem?id=2185 题意:就是要求一个字符矩阵的最小覆盖矩阵,可以在末尾不完全重合(即在末尾只要求最小覆盖矩阵的前缀覆盖剩余的尾部就行了) 分析: 先看 ...
 - POJ 2185 Milking Grid (KMP,求最小覆盖子矩阵,好题)
		
题意:给出一个大矩阵,求最小覆盖矩阵,大矩阵可由这个小矩阵拼成.(就如同拼磁砖,允许最后有残缺) 正确解法的参考链接:http://poj.org/showmessage?message_id=153 ...
 - POJ3080 Blue Jeans 题解 KMP算法
		
题目链接:http://poj.org/problem?id=3080 题目大意:给你N个长度为60的字符串(N<=10),求他们的最长公共子串(长度>=3). 题目分析:KMP字符串匹配 ...
 - HDU1711 Number Sequence 题解 KMP算法
		
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1711 题目大意:最基础的字符串匹配,只不过这里用整数数组代替了字符串. 给你两个数组 \(a[1..N ...
 
随机推荐
- 洛谷P1082 同余方程 [2012NOIP提高组D2T1] [2017年6月计划 数论06]
			
P1082 同余方程 题目描述 求关于 x 的同余方程 ax ≡ 1 (mod b)的最小正整数解. 输入输出格式 输入格式: 输入只有一行,包含两个正整数 a, b,用一个空格隔开. 输出格式: 输 ...
 - JSON和JSONP,原来ajax引用这个来实现跨域访问的
			
由于Sencha Touch 2这种开发模式的特性,基本决定了它原生的数据交互行为几乎只能通过AJAX来实现. 当然了,通过调用强大的PhoneGap插件然后打包,你可以实现100%的Socket通讯 ...
 - Spring_代理
			
1.代理模式. 2.静态代理原理及实践. 3.动态代理原理及实践. 4.Spring AOP原理及实战. 静态代理原理及实践 package test.staticProxy; // 接口 publi ...
 - React map生成元素添加点击事件绑定this
			
问题 使用.map(function(Item)生成元素添加onClick事件:onClick={this.provinceChange.bind(this, "99")}时,前台 ...
 - redis是当前流行的nosql数据库
			
redis是当前流行的nosql数据库,很多网站都用它来做缓存,今天我们来安装并配置下redis 二.安装并配置redis 1.安装redis sudo apt-get install redis-s ...
 - 462. Minimum Moves to Equal Array Elements II
			
Given a non-empty integer array, find the minimum number of moves required to make all array element ...
 - koa2路由
			
注意:必须导出 文档地址:https://npm.taobao.org/package/koa-router 例: const router = require('koa-router')() rou ...
 - larbin终于编译完成
			
1.目前sourceforge不能下载,只能找mirror进行下载了: 2.一些头文件始终找不到,定位了一下,应该是makefile中设置有问题,把CXXFLAGS中的-I-去掉,即可编译成功:
 - day39-Spring 01-上次课内容回顾
 - docker 与host互传文件
			
docker 的cp命令可以从容器往外复制,也可以从本机复制的容器. docker cp 文件路径 容器id:/容器目录 docker help cp Usage: docker cp [OPT ...