【BZOJ-4692】Beautiful Spacing 二分答案 + 乱搞(DP?)

4692: Beautiful Spacing

Time Limit: 15 Sec  Memory Limit: 128 MB
Submit: 46  Solved: 21
[Submit][Status][Discuss]

Description

文章是一些单词组成的序列,单词由字母组成。你的任务是将一篇文章的单词填充到一个网格中,其中网格包含W列和足够多的行。为了布局之美,以下限制都需要满足。

1.文章中的文字需要按照原有的顺序放置。下图表示了将4个单词的文章“This is a pen”放入11列的网格正确和错误的例子。

2.在同一行的两个相邻单词之间要有至少一个空格。有时你会需要放置多于一个空格来满足其他限制。

3.一个单词必须被放置到连续的列中,一个格子只含一个字符。你不能将通过增加空格或换行将一个单词分成多个部分。

4.文章必须占据边缘两列的格子,即每行的第一个单词必须占用第一个格子,且除了最后一行之外的每行的最后一个单词必须占用最后一个格子。

文章拥有最美的布局时,每一行都不会有过多的连续空格,在下图的例子里只有最多2个连续空格,而第一个例子里有3个连续空格,所以下图的例子比第一个例子美观。给定文章的信息和列数,请你找到一个最优的布局,使得最长连续空格尽量短。

Input

输入包含多组测试数据。每组数据第一行包含两个正整数W和N,其中W表示列数(3≤W≤80000),N表示文章的单词数量(2≤N≤50000)。

第二行包含N个正整数,按照文章中单词的顺序给出每个单词的字符数量,对于第i个单词的字符数量xi,有1≤x_i≤(W-1)/2,从而这也使得答案一定存在。

输入以两个零作为结束。

Output

对于每组数据,输出一个正整数表示最长连续空格长度的最小值。

Sample Input

11 4
4 2 1 3
5 7
1 1 1 2 2 1 2
11 7
3 1 3 1 3 3 4
100 3
30 30 39
30 3
2 5 3
0 0

Sample Output

2
1
2
40
1

HINT

Source

鸣谢Tangjz提供试题

Solution

二分+乱搞

首先,要求间隔最大的最小,显然可以二分答案.然后判定;

得到最大的间隔x后,我们可以得到用每个单词为一行结尾时的之前的可行范围

然后我们考虑通过之前的答案转移过来,

满足能放开[l,r](算上中间的空格),以及空格<=x,这样我们用两个指针l,r来找即可

我也不知道这种方法算是什么?模拟?

Code

#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<cmath>
using namespace std;
inline int read()
{
    int x=0; char ch=getchar();
    while (ch<‘0‘ || ch>‘9‘) {ch=getchar();}
    while (ch>=‘0‘ && ch<=‘9‘) {x=x*10+ch-‘0‘; ch=getchar();}
    return x;
}
#define MAXN 100010
int W,N,sum[MAXN];
bool f[MAXN];
inline bool check(int x)
{
    int l,r,t; l=r=t=0;
    for (int i=2; i<=N; i++)
        {
            while (i-r>=2 && (W-(sum[i]-sum[r]))<=x*(i-r-1)) t+=f[r++];
            while (l<=r && W-(sum[i]-sum[l])<i-l-1) t-=f[l++];
            if (t) f[i]=1; else f[i]=0;
        }
    for (int i=0; i<=N; i++) if (f[i] && W-(sum[N]-sum[i])>=N-i-1)
        return 1;
    return 0;
}
int main()
{
    f[0]=1;
    int l,r,mid;
    while (W=read(),N=read())
        {
            if (!W && !N) break;
            for (int i=1; i<=N; i++) sum[i]=sum[i-1]+read();
            l=1,r=W;
            while (l<=r)
                {
                    mid=(l+r)>>1;
                    if (!check(mid)) l=mid+1; else r=mid-1;
                }
            printf("%d\n",l);
        }
    return 0;
}

压了半天常,结果还是输rank1  100ms.....不甘心....

时间: 2024-10-11 16:36:14

【BZOJ-4692】Beautiful Spacing 二分答案 + 乱搞(DP?)的相关文章

BZOJ 2280 Poi2011 Plot 二分答案+随机增量法

题目大意:给定n个点,要求分成m段,使每段最小覆盖圆半径的最大值最小 二分答案,然后验证的时候把点一个个塞进最小覆盖圆中,若半径超了就分成一块-- 等等你在跟我说不随机化的随机增量法? 好吧 那么对于一个点pos,我们要计算最大的bound满足[pos,bound]区间内的最小覆盖圆半径不超过二分的值 直接上二分是不可取的,因为我们要求m次,如果每次都验证一遍[1,n/2]直接就炸了 我们可以这么搞 首先判断[pos,pos+1-1]是否满足要求 然后判断[pos,pos+2-1]是否满足要求

BZOJ 1816: [Cqoi2010]扑克牌( 二分答案 )

二分答案.. 一开始二分的初始右边界太小了然后WA,最后一气之下把它改成了INF... ------------------------------------------------------------------------ #include<cstdio> #include<cstring> #include<algorithm> #include<iostream> #define rep( i , n ) for( int i = 0 ; i

BZOJ 2525 Poi2011 Dynamite 二分答案+树形贪心

题目大意:给定一棵树,有一些点是关键点,要求选择不超过m个点,使得所有关键点到最近的选择的点距离最大值最小 二分答案,问题转化为: 给定一棵树,有一些点是关键点,要求选择最少的点使得每个关键点到选择的点的距离不超过limit 然后我们贪心DFS一遍 对于以一个节点为根的子树,有三种状态: 0.这棵子树中存在一个选择的点,这个选择的点的贡献还能继续向上传递 1.这棵子树中存在一个未被覆盖的关键点,需要一些选择的点去覆盖他 2.这棵子树中既没有能继续向上传递的选择的点也不存在未覆盖的关键点 是不是少

BZOJ 4077 Wf2014 Messenger 二分答案+计算几何

题目大意:给定两条折线,Alice沿着第一条折线走,Bob沿着第二条折线走,邮递员从Alice路径上的任意一点出发,沿直线走到Bob的路径上后刚好和Bob相遇,三人的速度都是1m/s,求邮递员走的最短距离,无解输出impossible 二分答案,然后让Bob提前出发mid,然后求出Alice和Bob全程的最短距离,判断是否≤mid就行了 无解比较难办,反正我是提前判断了无解的情况 #include <cmath> #include <cstdio> #include <cst

BZOJ 2792 Poi2012 Well 二分答案

题目大意:给定一个非负整数序列A,每次操作可以选择一个数然后减掉1,要求进行不超过m次操作使得存在一个Ak=0且max{Ai?Ai+1}最小,输出这个最小值以及此时最小的k 二分答案,然后验证的时候首先让相邻的都不超过x,然后枚举哪个点应该改成0 如果某个点需要改成0,那么需要进行操作的位置是一段区间,左右端点都单调,扫两遍就行了 #include <cstdio> int main() { cout<<"BZ炸掉了代码一会再贴吧233"<<endl

BZOJ 2654 tree(二分答案+并查集)

[题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=2654 [题目大意] 给你一个无向带权连通图,每条边是黑色或白色. 让你求一棵最小权的恰好有need条白色边的生成树.题目保证有解. [题解] 我们发现对于选中的边白色是从小到大的,黑色也是从小到大的, 因此我们对所有的白色边加一个权值,那么排序后做mst选取的白色边数量增减性单调, 对于增加的权值进行二分,验证能否满足要求即可. [代码] #include <cstdio> #in

BZOJ 2095 Poi2010 Bridges 二分答案+网络流

题目大意:给定一张图,每条边的两个方向有两个不同的权值,现在要求从1号节点出发遍历每条边一次且仅一次,最后回到1号节点,求最大边权的最小值 谁TM翻译的这道题给我滚出来看我不打死你 二分最大边的权值,然后就是经典的判断混合图欧拉回路存在性的问题了 #include <cstdio> #include <cstring> #include <iostream> #include <algorithm> #define M 2020 #define S 0 #d

uva live 6190 Beautiful Spacing (二分+dp检验 根据特有性质优化)

I - Beautiful Spacing Time Limit:8000MS     Memory Limit:65536KB     64bit IO Format:%lld & %llu Submit Status Description Text is a sequence of words, and a word consists of characters. Your task is to put words into a grid with W columns and suffic

BZOJ 1044 HAOI2008 木棍切割 二分答案+动态规划

题目大意:给定n个连在一起的木棍.分成m+1段.使每段最大值最小,求最大值的最小值及最大值最小时切割的方案数 第一问水爆了--二分答案妥妥秒过 第二问就有些难度了 首先我们令f[i][j]表示用前j个棒♂子得到i段的方案数 诶我没打什么奇怪的符号吧 于是我们有动规方程 f[i][j]=Σf[i-1][k] (sum[j]-sum[k]<=ans,k<j) 这个最坏情况下是O(m*n^2)的,肯定挂 我们发现k的下界是单调上升的 于是我们直接令k为当前j时k的下界.开一个变量记录k~j的f值之和