如果答案在某个碎片内部,那么直接悬线法解决,时间复杂度$O(n\sum)$。

如果$n$比较大,那么$\sum$比较小。

求出每个点向上能延伸的长度,枚举每个点向上这条线段作为短板。

算出完全可选的碎片的长度之和以及不能完全选,左边右边最大次大延伸距离,更新答案。

时间复杂度$O(n\sum^2)$。

如果$n$比较小,那么暴力枚举上下边界,计算答案方法同上。

时间复杂度$O(n^2\sum)$。

总时间复杂度$O(n\sum\sqrt{n\sum})$。

#include<cstdio>
const int N=100010,M=320;
int T,num,n,m,i,j,k,x,cnt,FL0,GL,FL1,FR0,GR,FR1,ans;
inline void up(int&f0,int&g0,int&f1,int x,int y){
if(x>f0){f1=f0,f0=x,g0=y;return;}
if(x>f1)f1=x;
}
inline void uans(int x){if(ans<x)ans=x;}
namespace NSMALL{
int st[N],en[N],f[N],g[N],w[N];char a[M][N],s[N];
void solve(){
for(i=1;i<=num;i++){
scanf("%d",&x);
st[i]=m+1;
en[i]=m+x;
for(j=1;j<=n;j++){
scanf("%s",s);
for(k=0;k<x;k++)a[j][k+st[i]]=s[k]-'0';
}
m+=x;
}
for(i=1;i<=m;i++)f[i]=1,g[i]=m,w[i]=0;
for(i=1;i<=n;i++){
for(GL=j=1;j<=m;j++)if(!a[i][j]){
w[j]++;
if(GL>f[j])f[j]=GL;
}else w[j]=0,f[j]=1,g[j]=m,GL=j+1;
for(GR=j=m;j;j--)if(!a[i][j]){
if(GR<g[j])g[j]=GR;
uans(w[j]*(g[j]-f[j]+1));
}else GR=j-1;
}
for(i=1;i<=n;i++){
for(k=1;k<=num;k++)f[k]=en[k],g[k]=st[k];
for(j=i;j<=n;j++){
cnt=FL0=GL=FL1=FR0=GR=FR1=0;
for(k=1;k<=num;k++){
for(x=st[k];x<=en[k];x++)if(a[j][x])break;
if(f[k]>x-1)f[k]=x-1;
for(x=en[k];x>=st[k];x--)if(a[j][x])break;
if(g[k]<x+1)g[k]=x+1;
if(f[k]==en[k]){cnt+=en[k]-st[k]+1;continue;}
up(FL0,GL,FL1,f[k]-st[k]+1,k);
up(FR0,GR,FR1,en[k]-g[k]+1,k);
}
if(GL!=GR)uans((j-i+1)*(cnt+FL0+FR0));else{
uans((j-i+1)*(cnt+FL0+FR1));
uans((j-i+1)*(cnt+FL1+FR0));
}
}
}
}
}
namespace NBIG{
int st[M],en[M],f[M],g[M],w[M];char a[N][M],s[M];
void solve(){
for(i=1;i<=num;i++){
scanf("%d",&x);
st[i]=m+1;
en[i]=m+x;
for(j=1;j<=n;j++){
scanf("%s",s);
for(k=0;k<x;k++)a[j][k+st[i]]=s[k]-'0';
}
m+=x;
}
for(i=1;i<=m;i++)f[i]=1,g[i]=m,w[i]=0;
for(i=1;i<=n;i++){
for(GL=j=1;j<=m;j++)if(!a[i][j]){
w[j]++;
if(GL>f[j])f[j]=GL;
}else w[j]=0,f[j]=1,g[j]=m,GL=j+1;
for(GR=j=m;j;j--)if(!a[i][j]){
if(GR<g[j])g[j]=GR;
uans(w[j]*(g[j]-f[j]+1));
}else GR=j-1;
}
for(i=1;i<=m;i++)w[i]=0;
for(i=1;i<=n;i++){
for(j=1;j<=m;j++)if(a[i][j])w[j]=0;else w[j]++;
for(j=1;j<=m;j++)if(w[j]){
cnt=FL0=GL=FL1=FR0=GR=FR1=0;
for(k=1;k<=num;k++){
for(x=st[k];x<=en[k];x++)if(w[x]<w[j])break;
f[k]=x-1;
for(x=en[k];x>=st[k];x--)if(w[x]<w[j])break;
g[k]=x+1;
if(f[k]==en[k]){cnt+=en[k]-st[k]+1;continue;}
up(FL0,GL,FL1,f[k]-st[k]+1,k);
up(FR0,GR,FR1,en[k]-g[k]+1,k);
}
if(GL!=GR)uans(w[j]*(cnt+FL0+FR0));else{
uans(w[j]*(cnt+FL0+FR1));
uans(w[j]*(cnt+FL1+FR0));
}
}
}
}
}
int main(){
scanf("%d",&T);
while(T--){
scanf("%d%d",&num,&n);
m=ans=0;
if(n<=315)NSMALL::solve();else NBIG::solve();
printf("%d\n",ans);
}
return 0;
}

  

BZOJ3873 : [Ahoi2014]拼图的更多相关文章

  1. BZOJ 3873: [Ahoi2014]拼图

    BZOJ 3873: [Ahoi2014]拼图 标签(空格分隔): OI-BZOJ OI-DP Time Limit: 10 Sec Memory Limit: 256 MB Description ...

  2. bzoj AC倒序

    Search GO 说明:输入题号直接进入相应题目,如需搜索含数字的题目,请在关键词前加单引号 Problem ID Title Source AC Submit Y 1000 A+B Problem ...

  3. 「AHOI2014/JSOI2014」拼图

    「AHOI2014/JSOI2014」拼图 传送门 看到 \(n \times m \le 10^5\) ,考虑根号分治. 对于 \(n < m\) 的情况,我们可以枚举最终矩形的上下边界 \( ...

  4. [AHOI2014/JSOI2014] 解题报告

    [AHOI2014/JSOI2014] 奇怪的计算器 一个很关键的结论,任何时候每个数的相对大小是不变的. 于是可以把这个相对大小当成线段树的权值,每次只需要维护一下区间极值和tag就好了,关于操作四 ...

  5. bzoj3874&2832 [Ahoi2014]宅男计划 模拟退火,三分

    [Ahoi2014&Jsoi2014]宅男计划 Time Limit: 1 Sec  Memory Limit: 256 MBSubmit: 962  Solved: 371[Submit][ ...

  6. [luogu] P4040 [AHOI2014/JSOI2014]宅男计划(贪心)

    P4040 [AHOI2014/JSOI2014]宅男计划 题目背景 自从迷上了拼图,JYY就变成了个彻底的宅男.为了解决温饱问题,JYY不得不依靠叫外卖来维持生计. 题目描述 外卖店一共有N种食物, ...

  7. Bzoj 3874: [Ahoi2014&Jsoi2014]宅男计划 三分+贪心

    3874: [Ahoi2014&Jsoi2014]宅男计划 Time Limit: 1 Sec  Memory Limit: 256 MBSubmit: 861  Solved: 336[Su ...

  8. 拼图小游戏之计算后样式与CSS动画的冲突

    先说结论: 前几天写了几个非常简单的移动端小游戏,其中一个拼图游戏让我郁闷了一段时间.因为要获取每张图片的位置,用`<style>`标签写的样式,直接获取计算后样式再用来交换位置,结果就悲 ...

  9. JavaScript版拼图小游戏

    慕课网上准备开个新的jQuery教程,花了3天空闲时间写了一个Javascript版的拼图小游戏,作为新教程配套的分析案例 拼图游戏网上有不少的实现案例了,但是此源码是我自己的实现,所以不做太多的比较 ...

随机推荐

  1. MVC学习笔记--跟小静学MVC相关语法特性小补习

    http://www.cnblogs.com/janes/archive/2012/10/15/2721101.html http://www.cnblogs.com/h82258652/p/4795 ...

  2. 菜鸟学Linux命令:lsof命令 查找指定用户、进程、端口打开的文件

    lsof,list open files, 是一个列出当前系统打开文件的工具.在linux环境下,任何事物都以文件的形式存在,通过文件不仅仅可以访问常规数据,还可以访问网络连接和硬件. 命令格式:ls ...

  3. Linux 标准目录结构

    初学Linux,首先需要弄清Linux 标准目录结构 / root --- 启动Linux时使用的一些核心文件.如操作系统内核.引导程序Grub等. home --- 存储普通用户的个人文件 ftp ...

  4. Delphi ini文件读写

    参考:http://www.cnblogs.com/zhangzhifeng/archive/2011/12/01/2270267.html 一.ini文件的结构 ;这是关于 ini 文件的注释 [节 ...

  5. 几年前做家教写的C教程(之一)

    C语言学习宝典 首先让我们认识什么是C语言. C语言是一种计算机开发语言,是一种非常基础的开发语言.能够用C语言做很多事情.C语言是顺序执行的程序. 程序应该包括数据描述,数据操作. C语言的数据类型 ...

  6. HP SAN Switch光纖交換機命令行畫zone

    有時候我們無法登陸網頁交互界面去操縱交換機,如下提供了命令行方式從交換機劃zone 1.創建別名 alicreate "SummaryDB_N", "211,14; 21 ...

  7. emc 郵件設置

    1. 進入Data Domain管理界面后,在Administration--->Settings界面.點擊More mail Server--->Set Mail Server---&g ...

  8. hdu 2897 巴什博弈变形 ***

    大意:一堆石子共有n个,A,B两人轮流从中取,每次取的石子数必须在[p,q]区间内,若剩下的石子数少于p个,当前取者必须全部取完.最后取石子的人输.给出n,p,q,问先取者是否有必胜策略? Bash博 ...

  9. hdu 4762 公式 java

    n/(n)^(m-1) import java.io.*; import java.math.*; import java.util.*; public class Main { static Big ...

  10. linux 读写锁应用实例

    转自:http://blog.csdn.net/dsg333/article/details/22113489 /*使用读写锁实现四个线程读写一段程序的实例,共创建了四个新的线程,其中两个线程用来读取 ...