一、题意

给定一个高度为$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)$。

三、代码

  1. #include<bits/stdc++.h>
  2. using namespace std;
  3. #define MAXN 1010
  4. char s[MAXN][MAXN];
  5. int n, m, R[MAXN][MAXN], D[MAXN][MAXN];
  6. int main() {
  7. // freopen("input.txt", "r", stdin);
  8. // freopen("output2.txt", "w", stdout);
  9. scanf("%d%d", &n, &m);
  10. ; i <= n; ++i) {
  11. scanf();
  12. ; j <= m; ++j) {
  13. ;
  14. ;
  15. }
  16. }
  17. ; i <= n; ++i) {
  18. ; j <= m; ++j) {
  19. , m), k = j;
  20. ;
  21. for(k = j; k <= r; ++k) {
  22. ))b |= 1LL << s[i][k];
  23. else {
  24. R[i][j] = k - ;
  25. break;
  26. }
  27. } )R[i][j] = k - ;
  28. }
  29. }
  30. ; j <= m; ++j) {
  31. ; i <= n; ++i) {
  32. , n), k = i;
  33. ;
  34. for(k = i; k <= r; ++k) {
  35. ))b |= 1LL << s[k][j];
  36. else {
  37. D[i][j] = k - ;
  38. break;
  39. }
  40. } )D[i][j] = k - ;
  41. }
  42. }
  43. ;
  44. ];
  45. ; i <= n; ++i) {
  46. ; j <= m; ++j) {
  47. , min(R[i][j], m)), k = j, minr;
  48. mind[] = D[i][j];
  49. ; k <= r; ++k) {
  50. mind[k - j] = min(mind[k - j - ], D[i][k]);
  51. }
  52. r = min(i + , min(D[i][j], n)), k = i, minr = R[i][j];
  53. for(k = i; k <= r; ++k) {
  54. while(minr > j && (R[k][j] < minr || mind[minr - j] < k))minr--;
  55. ans += minr - j + ;
  56. }
  57. }
  58. }
  59. cout << ans << '\n';
  60. ;
  61. }

牛客网暑期ACM多校训练营(第七场)J题(Sudoku Subrectangles)题解的更多相关文章

  1. 牛客网暑期ACM多校训练营(第二场)J farm (二维树状数组)

    题目链接: https://www.nowcoder.com/acm/contest/140/J 思路: 都写在代码注释里了,非常好懂.. for_each函数可以去看一下,遍历起vector数组比较 ...

  2. 牛客网 暑期ACM多校训练营(第二场)A.run-动态规划 or 递推?

    牛客网暑期ACM多校训练营(第二场) 水博客. A.run 题意就是一个人一秒可以走1步或者跑K步,不能连续跑2秒,他从0开始移动,移动到[L,R]的某一点就可以结束.问一共有多少种移动的方式. 个人 ...

  3. 牛客网 暑期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 ,问你 ...

  4. 2018牛客网暑期ACM多校训练营(第二场)I- car ( 思维)

    2018牛客网暑期ACM多校训练营(第二场)I- car 链接:https://ac.nowcoder.com/acm/contest/140/I来源:牛客网 时间限制:C/C++ 1秒,其他语言2秒 ...

  5. 牛客网暑期ACM多校训练营(第一场) - J Different Integers(线段数组or莫队)

    链接:https://www.nowcoder.com/acm/contest/139/J来源:牛客网 时间限制:C/C++ 2秒,其他语言4秒 空间限制:C/C++ 524288K,其他语言1048 ...

  6. 牛客网暑期ACM多校训练营(第九场) A题 FWT

    链接:https://www.nowcoder.com/acm/contest/147/A来源:牛客网 Niuniu has recently learned how to use Gaussian ...

  7. 牛客网暑期ACM多校训练营(第九场)D

    链接:https://www.nowcoder.com/acm/contest/147/D来源:牛客网 Niuniu likes traveling. Now he will travel on a ...

  8. 牛客网暑期ACM多校训练营(第二场)B discount

    链接:https://www.nowcoder.com/acm/contest/140/B来源:牛客网 题目描述 White Rabbit wants to buy some drinks from ...

  9. 2018牛客网暑期ACM多校训练营(第一场)D图同构,J

    链接:https://www.nowcoder.com/acm/contest/139/D来源:牛客网 同构图:假设G=(V,E)和G1=(V1,E1)是两个图,如果存在一个双射m:V→V1,使得对所 ...

  10. 牛客网暑期ACM多校训练营(第二场) I Car 思维

    链接:https://www.nowcoder.com/acm/contest/140/I来源:牛客网 White Cloud has a square of n*n from (1,1) to (n ...

随机推荐

  1. 使用 Koa + MongoDB + Redis 搭建论坛系统

    koa 相对于 express 的优势在于, 1.  使用了 yield generator 封装了co 框架, 使得异步处理, 能像同步那样书写 2.  使用了 中间件 ko-schema, 使得验 ...

  2. OC基础:数组.字典.集 分类: ios学习 OC 2015-06-18 18:58 47人阅读 评论(0) 收藏

    ==============NSArray(不可变数组)=========== NSArray,继承自NSObject  用来管理(储存)一些有序的对象,不可变数组. 创建一个空数组 NSArray ...

  3. mysql命令小结

    MySQL 数据库常用命令 1.MySQL常用命令 create database name; 创建数据库 use databasename; 选择数据库 drop database name 直接删 ...

  4. Oracle Statistic 统计信息 小结

    oraclestatisticstabledatabasesqldictionary   目录(?)[-] 直方图上列的信息说明 直方图类型说明   一.  Statistic 说明 Oracle 官 ...

  5. 多名历年高考状元最终选择做中国顶尖黑客 - imsoft.cnblogs

    原标题:揭秘中国顶尖黑客:二分之一是历年高考[微博]状元 “二分之一是各地的历年高考状元,二分之一是数学专业,二分之一来自微软.” Keen这样的团队是坚决与黑产划清界限的,这也是圈内的“道德洁癖”, ...

  6. 只需 5 秒钟,你就能取到 WPF 程序的超高分辨率超高清截图

    我想要截取一个 WPF 程序的图标,但是它太小了.如果我就这样截屏截下来,是很不高清的.由于我需要制作一份课件,所以我需要超高清版本,可是,如何做才能最快速拿到 WPF 程序的超高清截图呢? 本文分享 ...

  7. About DOM

    文档对象模型,简称DOM.是HTML页面结构给javascript提供的一个入口,它就像一个桥梁,连接两者的关系.通过DOM,JavaScript拥有访问并修改HTML的权限. 一.DOM模型: 首先 ...

  8. 【java规则引擎】《Drools7.0.0.Final规则引擎教程》第4章 4.2 activation-group& dialect& date-effective

    转载至:https://blog.csdn.net/wo541075754/article/details/75511887 activation-group 该属性将若干个规则划分成一个组,统一命名 ...

  9. 通过Jersey客户端API调用REST风格的Web服务

    Jersey 客户端 API 基础 要开始使用 Jersey 客户端 API,你首先需要创建一个 com.sun.jersey .api.client.Client 类的实例.下面是最简单的方法: i ...

  10. MySQL Disk--SSD磁盘性能抖动问题

    ============================================================= SSD性能 空盘性能:SSD出厂时磁盘没有任何数据情况下的性能 稳态性能:当 ...