牛客网暑期ACM多校训练营(第七场)J题(Sudoku Subrectangles)题解
一、题意
给定一个高度为$n$,宽度为$m$的字母矩形(有大写和小写字母,共$52$种),问里面有多少个子矩形是“数独矩形”。数独矩形是指,该矩形内每一行内元素都不相同,每一列内元素也不相同。
二、思路
对于每一个点$(i, j)$,预处理出$R[i][j]$,表示在第$i$行从第$j$列开始,往右一直到$R[i][j]$位置,这一段内所有元素都不相同。同理,再预处理出$D[i][j]$,表示在第$j$列从第$i$行开始,往下一直到$D[i][j]$位置,这一段内所有元素都不相同。显然,无论是往右还是往下这样的一段,长度都不会超过$52$。形式表示就是,$max\{R[i][j]\}=52(1 \le i \le n,1 \le j \le m)$,$max\{D[i][j]\}=52(1 \le i \le n,1 \le j \le m)$。
所以这部分的ACM版时间复杂度为$O(52*n*m)$。
接下来,枚举每一个点$(i, j)$,统计以点$(i, j)$作为左上角的数独矩形的个数。记住,一定要以点$(i, j)$作为左上角。
示意图如下,其中,绿色竖条代表$D[i][k](j \le k \le R[i][j])$,黄色横点条代表$R[k][j](i \le k \le D[i][j])$。
从$i$开始枚举每一行(显然只需要枚举到$D[i][j]$即可,最多循环$52$次),从第$i$行开始,记录一个右边界minr,表示:所有以$(i, j)$为左上角的行数为$k-i+1$的数独矩形最右边可到达的边界。假设当前在第$k$行,当前的右边界为$minr$,那么,所有以$(i, j)$为左上角的行数为$k-i+1$的数独矩形的个数为$minr-j+1$。举个例子,对于上图中,假设左上角坐标为$(i, j)$,当前枚举的行号$k=i$,则$minr=j+7$,这一行中数独矩形的个数为$minr-j+1=8$个,也就是$[(i, j)], [(i, j), (i, j+1)], [(i, j), (i, j+1), (i, j+2)], \dots, [(i, j), (i, j+1), (i, j+2), \dots, (i, minr)]$。
当枚举的行号$k=i+1$时,此时$minr=j+5=R[k][j]$,行数为两行(第$i$行和第$i+1$行)中数独矩形的个数为$minr-j+1=6$个。也就是:
后面的同理。然后,要注意的是,当$min\{D[i][k]\}(j \le k \le minr) < k$($j$~$minr$之间的最短的绿色竖条到不了$k$)或者$R[k][j] < minr$(第$k$行的黄色横点条到不了$minr$)时,说明以点$(i, j)$为左上角的高度为$k-i+1$的数独矩形的个数没有$minr-j+1$个,需要减小$minr$。
总的ACM版时间复杂度为$O(52*n*m)$。
三、代码
#include<bits/stdc++.h> using namespace std; #define MAXN 1010 char s[MAXN][MAXN]; int n, m, R[MAXN][MAXN], D[MAXN][MAXN]; int main() { // freopen("input.txt", "r", stdin); // freopen("output2.txt", "w", stdout); scanf("%d%d", &n, &m); ; i <= n; ++i) { scanf(); ; j <= m; ++j) { ; ; } } ; i <= n; ++i) { ; j <= m; ++j) { , m), k = j; ; for(k = j; k <= r; ++k) { ))b |= 1LL << s[i][k]; else { R[i][j] = k - ; break; } } )R[i][j] = k - ; } } ; j <= m; ++j) { ; i <= n; ++i) { , n), k = i; ; for(k = i; k <= r; ++k) { ))b |= 1LL << s[k][j]; else { D[i][j] = k - ; break; } } )D[i][j] = k - ; } } ; ]; ; i <= n; ++i) { ; j <= m; ++j) { , min(R[i][j], m)), k = j, minr; mind[] = D[i][j]; ; k <= r; ++k) { mind[k - j] = min(mind[k - j - ], D[i][k]); } r = min(i + , min(D[i][j], n)), k = i, minr = R[i][j]; for(k = i; k <= r; ++k) { while(minr > j && (R[k][j] < minr || mind[minr - j] < k))minr--; ans += minr - j + ; } } } cout << ans << '\n'; ; }
牛客网暑期ACM多校训练营(第七场)J题(Sudoku Subrectangles)题解的更多相关文章
- 牛客网暑期ACM多校训练营(第二场)J farm (二维树状数组)
题目链接: https://www.nowcoder.com/acm/contest/140/J 思路: 都写在代码注释里了,非常好懂.. for_each函数可以去看一下,遍历起vector数组比较 ...
- 牛客网 暑期ACM多校训练营(第二场)A.run-动态规划 or 递推?
牛客网暑期ACM多校训练营(第二场) 水博客. A.run 题意就是一个人一秒可以走1步或者跑K步,不能连续跑2秒,他从0开始移动,移动到[L,R]的某一点就可以结束.问一共有多少种移动的方式. 个人 ...
- 牛客网 暑期ACM多校训练营(第一场)A.Monotonic Matrix-矩阵转化为格子路径的非降路径计数,Lindström-Gessel-Viennot引理-组合数学
牛客网暑期ACM多校训练营(第一场) A.Monotonic Matrix 这个题就是给你一个n*m的矩阵,往里面填{0,1,2}这三种数,要求是Ai,j⩽Ai+1,j,Ai,j⩽Ai,j+1 ,问你 ...
- 2018牛客网暑期ACM多校训练营(第二场)I- car ( 思维)
2018牛客网暑期ACM多校训练营(第二场)I- car 链接:https://ac.nowcoder.com/acm/contest/140/I来源:牛客网 时间限制:C/C++ 1秒,其他语言2秒 ...
- 牛客网暑期ACM多校训练营(第一场) - J Different Integers(线段数组or莫队)
链接:https://www.nowcoder.com/acm/contest/139/J来源:牛客网 时间限制:C/C++ 2秒,其他语言4秒 空间限制:C/C++ 524288K,其他语言1048 ...
- 牛客网暑期ACM多校训练营(第九场) A题 FWT
链接:https://www.nowcoder.com/acm/contest/147/A来源:牛客网 Niuniu has recently learned how to use Gaussian ...
- 牛客网暑期ACM多校训练营(第九场)D
链接:https://www.nowcoder.com/acm/contest/147/D来源:牛客网 Niuniu likes traveling. Now he will travel on a ...
- 牛客网暑期ACM多校训练营(第二场)B discount
链接:https://www.nowcoder.com/acm/contest/140/B来源:牛客网 题目描述 White Rabbit wants to buy some drinks from ...
- 2018牛客网暑期ACM多校训练营(第一场)D图同构,J
链接:https://www.nowcoder.com/acm/contest/139/D来源:牛客网 同构图:假设G=(V,E)和G1=(V1,E1)是两个图,如果存在一个双射m:V→V1,使得对所 ...
- 牛客网暑期ACM多校训练营(第二场) I Car 思维
链接:https://www.nowcoder.com/acm/contest/140/I来源:牛客网 White Cloud has a square of n*n from (1,1) to (n ...
随机推荐
- utf-8和Unicode的区别
链接 utf-8和Unicode到底有什么区别?是存储方式不同?编码方式不同?它们看起来似乎很相似,但是实际上他们并不是同一个层次的概念 要想先讲清楚他们的区别,首先应该讲讲Unicode的来由. 众 ...
- Python学习(007)-函数的特性
一..函数是什么 定义: 函数是指将一组语句的集合通过一个名字(函数名)封装起来,要想执行这个函数,只需调用其函数名即可 特性: 1.代码重用 2.保持一致性 3.可扩展性 二.创建 2.1格式 de ...
- CSS 网格布局学习
转自:https://blog.jirengu.com/?p=990 CSS网格布局(又名“网格”)是一个二维的基于网格的布局系统,其目的只在于完全改变我们设计基于网格的用户界面的方式. CSS一直用 ...
- Hashtable实现原理及源码分析
Hashtable简介 和HashMap一样,Hashtable也是一个散列表,存储的内容是键值对(key-value)映射. Hashtable在Java中的定义为: public class Ha ...
- Linux下Bind error: Address already in use处理
发生这种问题是由于端口被程序绑定而没有释放造成. 可以使用netstat -lp查询当前处于连接的程序以及对应的进程信息. 如果只想查看指定端口,可以输入netstat -lp | grep 9877 ...
- 图解SQL的inner join、left join、right join、full outer join、union、union all的区别【转载】
对于SQL的Join,在学习起来可能是比较乱的.我们知道,SQL的Join语法有很多inner的,有outer的,有left的,有时候,对于Select出来的结果集是什么样子有点不是很清楚.Codin ...
- Android Kernel save defalut config
/********************************************************************************* * Android Kernel ...
- ZOJ 3551 吸血鬼 概率DP
解题报告链接: http://www.cnblogs.com/183zyz/archive/2012/09/13/2683524.html 做法:设当有i个吸血鬼时变成n个吸血鬼的天数的数学期望为dp ...
- dbt 基本试用
dbt 是一个很不错的进行etl 中的t 处理的工具,灵活简单,我们需要写的就是select 语句 dbt 帮助我们进行处理 测试集成了graphql 以及使用docker 运行 安装 pip ins ...
- RESTful 知识点
REST(英文:Representational State Transfer,简称REST) 对于资源的具体操作类型,由HTTP动词表示. 常用的HTTP动词有下面五个(括号里是对应的SQL命令). ...