Atcoder Grand Contest 033 D - Complexity(dp)
首先 \(n^5\) 的暴力非常容易想,设 \(dp_{a,b,c,d}\) 表示以 \((a,b)\) 为左上角,\((c,d)\) 为右下角的矩阵的 complexity。枚举断点转移即可,时间复杂度 \(n^5\)。
我们考虑优化这个 \(dp\),首先比较明显的一点:这个 \(dp\) 状态满足单调性,也就是说 \(\forall d_1<d_2,dp_{a,b,c,d_1}\le dp_{a,b,c,d_2}\),也就是说对于某个 \(dp\) 状态 \(dp_{a,b,c,d}\),假设最大的满足 \(dp_{a,b,c,k}<dp_{a,b,k+1,d}\) 的 \(k\) 为 \(r\),那么我们从行处切开的决策点 \(k\) 要么为 \(r\),要么为 \(r+1\),这个异常好想,于是我们可以二分这个 \(r\) 然后从 \(r\) 和 \(r+1\) 转移即可,时间复杂度 \(n^4\log n\),用双针也可优化到 \(n^4\)。
不过仅仅进行这个优化是远远不够的,我们还需进一步优化。这一步就有亿点考验观察力了,不难发现这个答案不会太大,手玩几组数据可看出这个答案最大不过 \(\log n\) 级别(当然如果硬要说理的话用二维线段树的思想也可以解释),也就是说我们 \(dp\) 数组的值域最大大概在 \(16\) 左右。所以我们可以很自然地想到一个类似于函数里面“交换定义域和值域”的思路:将答案放入 \(dp\) 状态中。
我们重新设计 \(dp\) 状态。设 \(f_{x,l,r,c}\) 表示最大的 \(y\) 满足 \(dp_{x,y,l,r}\leq c\)。转移分两种情况,横着切或者竖着切。如果我们横着切,那么我们肯定会贪心地选择最大的 \(dp_{x,y,l,r}\leq c-1\) 的 \(y\) 作为分割点,也就是说 \(f_{x,l,r,c}\leftarrow f_{f_{x,l,r,c-1}+1,l,r,c}\)。如果我们竖着切,考虑枚举断点 \(k\) 并从 \(k\) 与 \(k+1\) 之间切开,那么 \(dp_{x,y,l,r}\leq c\) 就意味着 \(dp_{x,y,l,k}\leq c-1,dp_{x,y,k+1,r}\leq c-1\),故我们用 \(\min(f_{x,l,k,c-1},f_{x,k+1,r,c-1})\) 更新 \(f_{x,l,r,c}\)。这样暴力更新还是 \(n^4\log n\) 的。不过还是按照之前的套路,由 \(dp\) 数组满足单调性也可推出 \(f\) 数组也满足单调性,即 \(\forall r_1<r_2,f_{x,l,r_1,c}\ge f_{x,l,r_2,c}\),于是我们二分断点转移即可,复杂度就降到了 \(n^3\log^2n\)。
#include <bits/stdc++.h>
using namespace std;
#define fi first
#define se second
#define fill0(a) memset(a,0,sizeof(a))
#define fill1(a) memset(a,-1,sizeof(a))
#define fillbig(a) memset(a,63,sizeof(a))
#define pb push_back
#define ppb pop_back
#define mp make_pair
template<typename T1,typename T2> void chkmin(T1 &x,T2 y){if(x>y) x=y;}
template<typename T1,typename T2> void chkmax(T1 &x,T2 y){if(x<y) x=y;}
typedef pair<int,int> pii;
typedef long long ll;
typedef unsigned int u32;
typedef unsigned long long u64;
namespace fastio{
#define FILE_SIZE 1<<23
char rbuf[FILE_SIZE],*p1=rbuf,*p2=rbuf,wbuf[FILE_SIZE],*p3=wbuf;
inline char getc(){return p1==p2&&(p2=(p1=rbuf)+fread(rbuf,1,FILE_SIZE,stdin),p1==p2)?-1:*p1++;}
inline void putc(char x){(*p3++=x);}
template<typename T> void read(T &x){
x=0;char c=getchar();T neg=0;
while(!isdigit(c)) neg|=!(c^'-'),c=getchar();
while(isdigit(c)) x=(x<<3)+(x<<1)+(c^48),c=getchar();
if(neg) x=(~x)+1;
}
template<typename T> void recursive_print(T x){if(!x) return;recursive_print(x/10);putc(x%10^48);}
template<typename T> void print(T x){if(!x) putc('0');if(x<0) putc('-'),x=~x+1;recursive_print(x);}
void print_final(){fwrite(wbuf,1,p3-wbuf,stdout);}
}
const int MAXN=185;
const int MAXANS=17;
int n,m,sum[MAXN+5][MAXN+5];char s[MAXN+5][MAXN+5];
int f[MAXN+5][MAXN+5][MAXN+5][MAXANS+1];
int main(){
scanf("%d%d",&n,&m);
for(int i=1;i<=n;i++) scanf("%s",s[i]+1);
for(int i=1;i<=n;i++) for(int j=1;j<=m;j++){
sum[i][j]=sum[i-1][j]+sum[i][j-1]-sum[i-1][j-1]+(s[i][j]=='.');
}
for(int i=1;i<=n;i++) for(int j=1;j<=m;j++) for(int k=j;k<=m;k++){
int l=i,r=n,p=i-1;
while(l<=r){
int mid=l+r>>1,ss=sum[mid][k]-sum[mid][j-1]-sum[i-1][k]+sum[i-1][j-1];
if(ss==0||ss==(mid-i+1)*(k-j+1)) p=mid,l=mid+1;
else r=mid-1;
} f[i][j][k][0]=p;
}
for(int p=1;p<=MAXANS;p++) for(int i=1;i<=n;i++) for(int len=1;len<=m;len++)
for(int j1=1,j2=len;j2<=m;j1++,j2++){
f[i][j1][j2][p]=f[i][j1][j2][p-1];
chkmax(f[i][j1][j2][p],f[f[i][j1][j2][p-1]+1][j1][j2][p-1]);
if(f[i][j1][j2][p]==n) continue;
int l=j1,r=j2-1;
while(l<=r){
int mid=l+r>>1;
chkmax(f[i][j1][j2][p],min(f[i][j1][mid][p-1],f[i][mid+1][j2][p-1]));
if(f[i][j1][mid][p-1]<f[i][mid+1][j2][p-1]) r=mid-1;
else l=mid+1;
}
}
int ans=-1;
for(int i=0;i<=MAXANS;i++) if(f[1][1][m][i]<n) chkmax(ans,i);
printf("%d\n",ans+1);
return 0;
}
Atcoder Grand Contest 033 D - Complexity(dp)的更多相关文章
- AtCoder Grand Contest 031 B - Reversi(DP)
B - Reversi 题目链接:https://atcoder.jp/contests/agc031/tasks/agc031_b 题意: 给出n个数,然后现在你可以对一段区间修改成相同的值,前提是 ...
- Atcoder Grand Contest 021 F - Trinity(dp+NTT)
Atcoder 题面传送门 & 洛谷题面传送门 首先我们考虑设 \(dp_{i,j}\) 表示对于一个 \(i\times j\) 的网格,其每行都至少有一个黑格的合法的三元组 \((A,B, ...
- Atcoder Regular Contest 089 D - ColoringBalls(DP)
Atcoder 题面传送门 & 洛谷题面传送门 神仙题. 在下文中,方便起见,用 R/B 表示颜色序列中球的颜色,用 r/b 表示染色序列中将连续的区间染成的颜色. 首先碰到这一类计算有多少个 ...
- AtCoder Grand Contest 032-B - Balanced Neighbors (构造)
Time Limit: 2 sec / Memory Limit: 1024 MB Score : 700700 points Problem Statement You are given an i ...
- AtCoder Grand Contest 012 B - Splatter Painting(dp)
Time limit : 2sec / Memory limit : 256MB Score : 700 points Problem Statement Squid loves painting v ...
- Atcoder Grand Contest 001E - BBQ Hard(组合意义转化,思维题)
Atcoder 题面传送门 & 洛谷题面传送门 Yet another 思维题-- 注意到此题 \(n\) 数据范围很大,但是 \(a_i,b_i\) 数据范围很小,这能给我们什么启发呢? 观 ...
- AtCoder Grand Contest 033
为什么ABC那么多?建议Atcoder多出些ARC/AGC,好不容易才轮到AGC…… A 签到.就是以黑点为源点做多元最短路,由于边长是1直接bfs就好了,求最长路径. #include<bit ...
- Atcoder Grand Contest 038 E - Gachapon(Min-Max 容斥+背包)
Atcoder 题面传送门 & 洛谷题面传送门 我竟然能独立做出 Ag 的 AGC E,incredible!更新了 Atcoder 做题难度上限( 首先按照套路 Min-Max 容斥,\(a ...
- AtCoder Grand Contest 033 题解
传送门 我比赛的时候怕不是在睡觉啊-- \(A\ Darker\ and\ Darker\) 我是不是想得太复杂了--根本没必要像我这样做吧-- 首先问题可以转化成令\(p_{i,j}\)表示到\(( ...
随机推荐
- 从源码层面深度剖析Redisson实现分布式锁的原理(全程干货,注意收藏)
Redis实现分布式锁的原理 前面讲了Redis在实际业务场景中的应用,那么下面再来了解一下Redisson功能性场景的应用,也就是大家经常使用的分布式锁的实现场景. 引入redisson依赖 < ...
- FastAPI 学习之路(四十六)WebSockets(二)
上一篇文章,我们分享了WebSockets一些入门的,我们这节课,在原来的基础上,对于讲解的进行一个演示.我们最后分享了依赖token等.首先我们对上次的代码进行调整. 我们之前分享FastAPI 学 ...
- Java继承、重写与重载
1.java继承 1.1概念 继承就是子类继承父类的特征和行为,使得子类对象(实例)具有父类的实例域和方法,或子类从父类继承方法,使得子类具有父类相同的行为. 继承可以使用extends和implem ...
- sql递归查询部门数据
1 with cte as 2 ( 3 select a.DepartCode,a.DepartName,a.ParentDepartCode from tbDeparts a where Paren ...
- 【二食堂】Beta - Scrum Meeting 7
Scrum Meeting 7 例会时间:5.19 18:30~18:50 进度情况 组员 当前进度 今日任务 李健 1. 文本区域的前后端对接完成,bug已经修复issue2. 自定义关系的添加与删 ...
- Noip模拟54 2021.9.16
T1 选择 现在发现好多题目都是隐含的状压,不明面给到数据范围里,之凭借一句话 比如这道题就是按照题目里边给的儿子数量不超过$10$做状压,非常邪门 由于数据范围比较小,怎么暴力就怎么来 从叶子节点向 ...
- c语言编程基础入门必备知识
数据类型 基本数据类型 类型名称说明char字符类型存放字符的ASCII码int整型存放有符号整数short短整型存放有符号整数long长整型存放有符号整数long long存放有符号整数float单 ...
- arduino 使用 analogRead 读取不到数据,digitalRead 却可以正常读取
项目场景: 最近在使用安信可的 ESP32S P14 引脚(ADC 16)读取一个电路状态的时候遇到一个问题,电路状态不是很稳定,在高电平的时候,会突然出现毫秒级的波动,出现短暂的低电平,造成设备状态 ...
- 设置IDEA启动,不要自动打开上次使用时的项目
打开idea时自动加载最近编辑的项目,很费时间,关闭设置如下
- 【SVG】为了前端页面的美丽,我选择学习SVG
[SVG]为了前端页面的美丽,我选择学习SVG 博客说明 文章所涉及的资料来自互联网整理和个人总结,意在于个人学习和经验汇总,如有什么地方侵权,请联系本人删除,谢谢! 说明 SVG在之前自学的过程中, ...