https://nanti.jisuanke.com/t/30991
题意
一个n*m的方格矩阵,有的格子被涂成了黑色,问该矩阵中有多少个子矩阵,子矩阵不包含黑色格子。
分析
参考https://blog.****.net/Sirius_han/article/details/82313029
n=1e5,m=100。首先思考一下没有黑点的话,子矩阵总数怎么算。
现有长为L的大矩阵,对于固定高度h,其子矩阵的个数是这样计算的
for(int i=; i<=L; i++){
for(int j=i; j>; j--){
count+=h;
}
}
那么对于不同高度,只需要再加一维循环就好。
解决有黑点的问题,当存在黑点时
先看高为H(4)的子矩阵个数:以(4, 7)为右下角的高为H的子矩阵个数为3个,由L=4处再向左,就只能构成高为2的子矩阵了;
那么怎么该上边的代码才能得出答案呢?如下:
for(int i=; i<=H; i++){
for(int j=; j<=L; j++){
h=i;
for(int k=j; k>; k--){
h=min(h, i-p[k]);
count+=h;
}
}
}
//p[k]表示第k列中在i行上边的第一个黑点的位置,
那么维护p数组就是这题的重点了
#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
int b[][], up[];
int main(){
int T, cas=;
scanf("%d", &T);
while(T--){
int n, m, K;
scanf("%d%d%d", &n, &m, &K);
for(int i=; i<=n; i++){
for(int j=; j<=m; j++){
b[i][j]=;
up[j]=;
}
}
for(int i=; i<K; i++){
int x, y;
scanf("%d%d", &x, &y);
b[x][y]=;
}
ll ans=;
for(int i=; i<=n; i++){
for(int j=; j<=m; j++){
if(b[i][j]){
up[j]=i;
}
}
for(int j=; j<=m; j++){
ll minn=0x7f7f7f7f7f7f7f7f;
for(int k=j; k>; k--){
minn=min(minn, (ll)(i-up[k]));
ans+=minn;
}
}
}
printf("Case #%d: %lld\n", ++cas, ans);
}
return ;
}
- #include <bits/stdc++.h>
-
using namespace std;
-
typedef long long ll;
-
int b[100010][110], up[110];
-
int main(){
-
int T, cas=0;
-
scanf("%d", &T);
-
while(T--){
-
int n, m, K;
-
scanf("%d%d%d", &n, &m, &K);
-
for(int i=0; i<=n; i++){
-
for(int j=0; j<=m; j++){
-
b[i][j]=0;
-
up[j]=0;
-
}
-
}
-
for(int i=0; i<K; i++){
-
int x, y;
-
scanf("%d%d", &x, &y);
-
b[x][y]=1;
-
}
-
ll ans=0;
-
for(int i=1; i<=n; i++){
-
for(int j=1; j<=m; j++){
-
if(b[i][j]){
-
up[j]=i;
-
}
-
}
-
for(int j=1; j<=m; j++){
-
ll minn=0x7f7f7f7f7f7f7f7f;
-
for(int k=j; k>0; k--){
-
minn=min(minn, (ll)(i-up[k]));
-
ans+=minn;
-
}
-
}
-
}
-
printf("Case #%d: %lld\n", ++cas, ans);
-
}
-
return 0;
-
}