Solution -「Gym 102956F」Border Similarity Undertaking
\(\mathcal{Description}\)
Link.
给定一张 \(n\times m\) 的表格,每个格子上写有一个小写字母。求其中长宽至少为 \(2\),且边界格子上字母相同的矩形个数。
\(n,m\le2\times10^3\)。
\(\mathcal{Solution}\)
可以感知到这是道分治题。
不妨设当前处理左上角 \((u,l)\),右下角 \((d,r)\) 的矩形内的所有答案,且 \(d-u>r-l\)。那么取行的一半 \(p=\lfloor\frac{u+d}{2}\rfloor\),尝试求出所有在矩形内且跨过 \(p\) 这条水平直线的矩形数量。对于 \(p\) 上的每个点 \((p,i)\),预处理出其 向上/向下 走到的同种格子中,有多少个能 向左/向右 走 \(x\) 步,然后枚举矩形跨过 \(p\) 的两个位置 \(i,j\),讨论 \(i,j\) 向上/向下 能走步数的大小关系,利用预处理的信息计算答案。
这样每层做到 \(\mathcal O((d-u)(r-l))\),所以总复杂度 \(\mathcal O(nm(\log n+\log m))\)。
\(\mathcal{Code}\)
/*~Rainybunny~*/
#include <bits/stdc++.h>
#define rep( i, l, r ) for ( int i = l, rep##i = r; i <= rep##i; ++i )
#define per( i, r, l ) for ( int i = r, per##i = l; i >= per##i; --i )
typedef long long LL;
inline int imin( const int a, const int b ) { return a < b ? a : b; }
inline int imax( const int a, const int b ) { return a < b ? b : a; }
const int MAXN = 2e3;
int n, m;
int up[MAXN + 5][MAXN + 5], dn[MAXN + 5][MAXN + 5];
int le[MAXN + 5][MAXN + 5], ri[MAXN + 5][MAXN + 5];
int sum[MAXN + 5][MAXN + 5][4];
char grid[MAXN + 5][MAXN + 5];
LL ans;
inline void solve( const int u, const int d, const int l, const int r ) {
if ( u + 1 > d || l + 1 > r ) return ;
if ( d - u > r - l ) {
int mr = u + d >> 1, lr = r - l + 1;
solve( u, mr, l, r ), solve( mr + 1, d, l, r );
rep ( i, l, r ) rep ( j, 0, lr ) {
sum[i][j][0] = sum[i][j][1] = sum[i][j][2] = sum[i][j][3] = 0;
}
rep ( i, l, r ) {
rep ( j, imax( u, mr - up[mr][i] + 1 ),
imin( d, mr + dn[mr][i] - 1 ) ) {
++sum[i][imin( ri[j][i], lr )][( mr < j ) * 2];
++sum[i][imin( le[j][i], lr )][( mr < j ) * 2 + 1];
}
per ( j, lr - 1, 0 ) rep ( k, 0, 3 ) {
sum[i][j][k] += sum[i][j + 1][k];
}
}
rep ( i, l, r ) rep ( j, i + 1, r ) {
if ( grid[mr][i] == grid[mr][j] ) {
ans += 1ll *
( up[mr][i] < up[mr][j] ?
sum[i][j - i + 1][0] : sum[j][j - i + 1][1] ) *
( dn[mr][i] < dn[mr][j] ?
sum[i][j - i + 1][2] : sum[j][j - i + 1][3] );
}
}
} else {
int mc = l + r >> 1, ud = d - u + 1;
solve( u, d, l, mc ), solve( u, d, mc + 1, r );
rep ( i, u, d ) rep ( j, 0, ud ) {
sum[i][j][0] = sum[i][j][1] = sum[i][j][2] = sum[i][j][3] = 0;
}
rep ( i, u, d ) {
rep ( j, imax( l, mc - le[i][mc] + 1 ),
imin( r, mc + ri[i][mc] - 1 ) ) {
++sum[i][imin( dn[i][j], ud )][( mc < j ) * 2];
++sum[i][imin( up[i][j], ud )][( mc < j ) * 2 + 1];
}
per ( j, ud - 1, 0 ) rep ( k, 0, 3 ) {
sum[i][j][k] += sum[i][j + 1][k];
}
}
rep ( i, u, d ) rep ( j, i + 1, d ) {
if ( grid[i][mc] == grid[j][mc] ) {
ans += 1ll *
( le[i][mc] < le[j][mc] ?
sum[i][j - i + 1][0] : sum[j][j - i + 1][1] ) *
( ri[i][mc] < ri[j][mc] ?
sum[i][j - i + 1][2] : sum[j][j - i + 1][3] );
}
}
}
}
int main() {
scanf( "%d %d", &n, &m );
rep ( i, 1, n ) scanf( "%s", grid[i] + 1 );
rep ( i, 1, n ) rep ( j, 1, m ) {
le[i][j] = grid[i][j] == grid[i][j - 1] ? le[i][j - 1] + 1 : 1;
up[i][j] = grid[i][j] == grid[i - 1][j] ? up[i - 1][j] + 1 : 1;
}
per ( i, n, 1 ) per ( j, m, 1 ) {
ri[i][j] = grid[i][j] == grid[i][j + 1] ? ri[i][j + 1] + 1 : 1;
dn[i][j] = grid[i][j] == grid[i + 1][j] ? dn[i + 1][j] + 1 : 1;
}
solve( 1, n, 1, m );
printf( "%lld\n", ans );
return 0;
}
Solution -「Gym 102956F」Border Similarity Undertaking的更多相关文章
- Solution -「Gym 102956F」Find the XOR
\(\mathcal{Description}\) Link. 给定 \(n\) 个点 \(m\) 条边的连通无向图 \(G\),边有边权.其中 \(u,v\) 的距离 \(d(u,v)\) ...
- Solution -「Gym 102979E」Expected Distance
\(\mathcal{Description}\) Link. 用给定的 \(\{a_{n-1}\},\{c_n\}\) 生成一棵含有 \(n\) 个点的树,其中 \(u\) 连向 \([1, ...
- Solution -「Gym 102979L」 Lights On The Road
\(\mathcal{Description}\) Link. 给定序列 \(\{w_n\}\),选择 \(i\) 位置的代价为 \(w_i\),要求每个位置要不被选择,要不左右两个位置至少被 ...
- Solution -「Gym 102956B」Beautiful Sequence Unraveling
\(\mathcal{Description}\) Link. 求长度为 \(n\),值域为 \([1,m]\) 的整数序列 \(\lang a_n\rang\) 的个数,满足 \(\not\ ...
- Solution -「Gym 102956A」Belarusian State University
\(\mathcal{Description}\) Link. 给定两个不超过 \(2^n-1\) 次的多项式 \(A,B\),对于第 \(i\in[0,n)\) 个二进制位,定义任意一个二元 ...
- Solution -「Gym 102798I」Sean the Cuber
\(\mathcal{Description}\) Link. 给定两个可还原的二阶魔方,求从其中一个状态拧到另一个状态的最小步数. 数据组数 \(T\le2.5\times10^5\). ...
- Solution -「Gym 102798K」Tree Tweaking
\(\mathcal{Description}\) Link. 给定排列 \(\{p_n\}\),求任意重排 \(p_{l..r}\) 的元素后,将 \(\{p_n\}\) 依次插入二叉搜索树 ...
- Solution -「Gym 102798E」So Many Possibilities...
\(\mathcal{Description}\) Link. 给定非负整数序列 \(\{a_n\}\) 和 \(m\),每次随机在 \(\{a\}\) 中取一个非零的 \(a_i\)(保证存 ...
- Solution -「Gym 102759I」Query On A Tree 17
\(\mathcal{Description}\) Link. 给定一棵含 \(n\) 个结点的树,结点 \(1\) 为根,点 \(u\) 初始有点权 \(a_u=0\),维护 \(q\) 次 ...
随机推荐
- linux 三剑客(持续更新)排版后续再说,边学边记笔记
切记:seq命令用于产生从某个数到另外一个数之间的所有整数.sed才是处理文本的命令 在遇到扩展符号时,需要添加特定参数,| () +[] 为扩展符号时,必须添加参数 egrep/grep -E s ...
- vant引入及配置
1. vant 官网 https://youzan.github.io/vant/#/zh-CN/quickstart 2. 通 npm 安装 npm i vant -S 3.安装 babel-plu ...
- linux下玩转磁盘管理与挂载硬盘
前言 本文将带来linux下的磁盘管理中的硬盘挂载,Linux操作系统挂载硬盘需要了解的一些知识.这可能是迄今为止介绍的最最最实用的linux硬盘挂载的文章了,比较详细.由于工作原因,平时使用的比较多 ...
- MCU软件最佳实践——独立按键
1. 引子 在进行mcu驱动和应用开发时,经常会遇到独立按键驱动的开发,独立按键似乎是每一个嵌入式工程师的入门必修课.笔者翻阅了许多书籍(包括上大学时候用的书籍)同时查阅了网上许多网友的博客,无一例外 ...
- golang gin框架中实现大文件的流式上传
一般来说,通过c.Request.FormFile()获取文件的时候,所有内容都全部读到了内存.如果是个巨大的文件,则可能内存会爆掉:且,有的时候我们需要一边上传一边处理. 以下的代码实现了大文件流式 ...
- django_url配置
前言 我们在浏览器访问一个网页是通过url地址去访问的,django管理url配置是在urls.py文件.当一个页面数据很多时候,通过会有翻页的情况,那么页数是不固定的,如:page=1.也就是url ...
- Anchor CMS 0.12.7 跨站请求伪造漏洞(CVE-2020-23342)
这个漏洞复现相对来说很简单,而且这个Anchor CMS也十分适合新手训练代码审计能力.里面是一个php框架的轻量级设计,通过路由实现的传递参数. 0x00 漏洞介绍 Anchor(CMS)是一款优秀 ...
- FHQtreap(我有个绝妙的理解方法,但课的时间不够[doge])
FHQtreap板子(P1486 [NOI2004] 郁闷的出纳员) 会了FHQ,treap什么的就忘了吧...... #include<bits/stdc++.h> using name ...
- golang中channel讲解
1. 无缓冲通道 2. 有缓冲通道 有缓冲通道特点:当channel已经满,在向里面写数据就会阻塞,当channel已经为空,在从里面读数据就会阻塞. 3. 关闭channel package mai ...
- 集合框架-工具类-Collection-toArray方法
1 package cn.itcast.p3.toolclass.arrays.demo; 2 3 import java.util.ArrayList; 4 import java.util.Arr ...