题目描述 Description

小G是一个出色的诗人,经常作诗自娱自乐。但是,他一直被一件事情所困扰,那就是诗的排版问题。

一首诗包含了若干个句子,对于一些连续的短句,可以将它们用空格隔开并放在一行中,注意一行中可以放的句子数目是没有限制的。小G给每首诗定义了一 个行标准长度(行的长度为一行中符号的总个数),他希望排版后每行的长度都和行标准长度相差不远。显然排版时,不应改变原有的句子顺序,并且小G不允许把 一个句子分在两行或者更多的行内。在满足上面两个条件的情况下,小G对于排版中的每行定义了一个不协调度, 为这行的实际长度与行标准长度差值绝对值的P次方,而一个排版的不协调度为所有行不协调度的总和。

小G最近又作了几首诗,现在请你对这首诗进行排版,使得排版后的诗尽量协调(即不协调度尽量小),并把排版的结果告诉他。

输入描述 Input Description

本题中包含多组测试数据。

输入文件中的第一行为一个整数T,表示诗的数量。

接下来为T首诗,这里一首诗即为一组测试数据。每组测试数据中的第一行为三个由空格分隔的正整数N,L,P,其中:N表示这首诗句子的数目,L表示这首诗的行标准长度,P的含义见问题描述。

从第二行开始,每行为一个句子,句子由英文字母、数字、标点符号等符号组成(ASCII码33~127,但不包含'-')。

输出描述 Output Description

对于每组测试数据,若最小的不协调度不超过10^18,则第一行为一个数,表示不协调度。接下来若干行,表示你排版之后的诗。注意:在同一行的相邻两个句子之间需要用一个空格分开。

如果有多个可行解,它们的不协调度都是最小值,则输出任意一个解均可。若最小的不协调度超过10^18,则输出“Too hard to arrange”(不含引号)。每组测试数据结束后输出“--------------------”(不含引号),共20个“-”,“-”的ASCII 码为45,请勿输出多余的空行或者空格。

由于缺少special judge,因此在这里只要求输出最小的不协调度。格式不变,依然以"-"分割。

样例输入 Sample Input

4

4 9 3

brysj,

hhrhl.

yqqlm,

gsycl.

4 9 2

brysj,

hhrhl.

yqqlm,

gsycl.

1 1005 6

poet

1 1004 6

poet

样例输出 Sample Output

108

--------------------

32

--------------------

Too hard to arrange

--------------------

1000000000000000000
--------------------

数据范围及提示 Data Size & Hint

【样例说明】

前两组输入数据中每行的实际长度均为6,后两组输入数据每行的实际长度均为4。一个排版方案中每行相邻两个句子之间的空格也算在这行的长度中(可参见样例中第二组数据)。每行末尾没有空格。

总共10个测试点,数据范围满足:

测试点

T

N

L

P

1

<=10

<=18

<=100

<=5

2

<=10

<=2000

<=60000

<=10

3

<=10

<=2000

<=60000

<=10

4

<=5

<=100000

<=200

<=10

5

<=5

<=100000

<=200

<=10

6

<=5

<=100000

<=3000000

2

7

<=5

<=100000

<=3000000

2

8

<=5

<=100000

<=3000000

<=10

9

<=5

<=100000

<=3000000

<=10

10

<=5

<=100000

<=3000000

<=10

所有测试点中均满足句子长度不超过30。

这个DP的模型跟玩具装箱几乎一模一样,都是划分型DP。。。

设f[i]表示到第i句话的最优值。。。

记一个前缀和a,  n^2 的转移 f[i]=min(f[i],f[j]+(abs(i-j-1+a[i]-a[j]-L)^p));30分

这题巨坑!!!乘会爆long long!!! 要开long double!!!

不开就只有10分。。。

这题有决策单调性(自己打表)

对与决策单调性有一个常数优化,即每次从上次最大的能转移的点开始枚举,这样有50分,但在某些题目中,用这个东西经常可以AC!!!

70分的话,在50--70的部分分打玩具装箱的斜率优化。。。

这题是p次方所以不能用斜率优化做。。。

正经的决策单调性的解决办法是什么呢。。。就是二分栈!!!

<<1D1D动态规划优化初步>>这篇文章说得很好。。。


使用一个栈来维护数据,栈中的每一个元素保存一个决策的起始位置与终了位置,显然这些位置相互连接且依次递增。

当插入一个新的决策时,从后到前扫描栈,对于每一个老决策来说,做这样两件事:

  1. 如果在老决策的起点处还是新决策更好,则退栈,全额抛弃老决策,将其区间合并至新决策中,继续扫描下一个决策。

  2. 如果在老决策的起点处是老决策好,则转折点必然在这个老决策的区间中;二分查找之,然后新决策进栈,结束。

由于一个决策出栈之后再也不会进入,所以均摊时间为O(1),但是由于二分查找的存在,所以整个算法的时间复杂度为O(nlogn)。

这题硬是要卡乘爆啊!!!WA了无数遍。。。实现参考hzwer。

 // MADE BY QT666
#include<cstdio>
#include<algorithm>
#include<cmath>
#include<iostream>
#include<queue>
#include<set>
#include<cstdlib>
#include<cstring>
#include<string>
#include<ctime>
#include<iomanip>
#define lson num<<1
#define rson num<<1|1
using namespace std;
const int N=;
int gi()
{
int x=,flag=;
char ch=getchar();
while(ch<''||ch>''){if(ch=='-') flag=-;ch=getchar();}
while(ch>=''&&ch<='') x=x*+ch-'',ch=getchar();
return x*flag;
}
char ch[];
long double a[N],f[N];
int n,L,p;
struct data{int l,r,p;}q[N];
long double work(long double x,int p){
long double ret=;
if(x<) x=-x;
for(int i=;i<=p;i++){
ret*=x;
}
return ret;
}
long double cal(int j,int i){
return f[j]+work(a[i]-a[j]+i-j--L,p);
}
int find(data t,int x)
{
int l=t.l,r=t.r;
while(l<=r)
{
int mid=(l+r)>>;
if(cal(t.p,mid)<cal(x,mid))
l=mid+;
else r=mid-;
}
return l;
}
main()
{
int T;
T=gi();
while(T--)
{
n=gi(),L=gi(),p=gi();
for(int i=;i<=n;i++){
scanf("%s",ch+);
a[i]=a[i-]+strlen(ch+);
}
/*int last=1;
for(int i=1;i<=n;i++){
f[i]=work((int)abs(i-1+a[i]-L),p);
for(int j=last;j<i;j++){
int y=f[j]+work((int)abs(i-j-1+a[i]-a[j]-L),p);
if(y<f[i]) f[i]=y,last=j;
}
}*/
int head=,tail=;
q[++tail]=(data){,n,};
for(int i=;i<=n;i++){
if(head<=tail&&i>q[head].r)head++;
f[i]=cal(q[head].p,i);
if(head>tail||cal(i,n)<=cal(q[tail].p,n)){
while(head<=tail&&cal(i,q[tail].l)<=cal(q[tail].p,q[tail].l))
tail--;
if(head>tail)
q[++tail]=(data){i,n,i};
else {
int t=find(q[tail],i);
q[tail].r=t-;
q[++tail]=(data){t,n,i};
}
}
}
if(f[n]>) puts("Too hard to arrange");
else printf("%lld\n",(long long)f[n]);
puts("--------------------");
}
return ;
}

NOI 2009 诗人小G的更多相关文章

  1. [BZOJ 1563] [NOI 2009] 诗人小G(决策单调性)

    [BZOJ 1563] [NOI 2009] 诗人小G(决策单调性) 题面 一首诗包含了若干个句子,对于一些连续的短句,可以将它们用空格隔开并放在一行中,注意一行中可以放的句子数目是没有限制的.小 G ...

  2. 解题:NOI 2009 诗人小G

    题面 今天考试考了,于是开始糊学决策单调性DP 这是一个完全不会优化DP的人 决策单调性DP的一种优化方法是用单调队列优化 存下{左端点l,右端点r,最优决策点p}的三元组,按照单调队列的通常操作来说 ...

  3. NOI 2009A 诗人小G

    NOI 2009A 诗人小G 诗人小G [问题描述] 小G是一个出色的诗人,经常作诗自娱自乐.但是,他一直被一件事情所困扰,那就是诗的排版问题. 一首诗包含了若干个句子,对于一些连续的短句,可以将它们 ...

  4. C++之路进阶——codevs2933(诗人小G)

    2933 诗人小G 2009年NOI全国竞赛  时间限制: 1 s  空间限制: 128000 KB  题目等级 : 大师 Master     题目描述 Description 小G是一个出色的诗人 ...

  5. 【Luogu1912】【NOI2009】诗人小G(动态规划)

    [Luogu1912][NOI2009]诗人小G(动态规划) 题面 洛谷 题解 原来\(NOI\)这么多神仙题... 考虑一个极其明显的\(dp\) 设\(f[i]\)表示前\(i\)个句子产生的最小 ...

  6. LG1912 [NOI2009]诗人小G

    题意 题目描述 小G是一个出色的诗人,经常作诗自娱自乐.但是,他一直被一件事情所困扰,那就是诗的排版问题. 一首诗包含了若干个句子,对于一些连续的短句,可以将它们用空格隔开并放在一行中,注意一行中可以 ...

  7. bzoj1563: [NOI2009]诗人小G 决策单调性(1D1D)

    目录 题目链接 题解 代码 题目链接 bzoj1563: [NOI2009]诗人小G 题解 \(n^2\) 的dp长这样 \(f_i = min(f_j + (sum_i - sum_j - 1 - ...

  8. 1563: [NOI2009]诗人小G

    1563: [NOI2009]诗人小G https://lydsy.com/JudgeOnline/problem.php?id=1563 分析: 直接转移f[i]=f[j]+cost(i,j),co ...

  9. BZOJ1563/洛谷P1912 诗人小G 【四边形不等式优化dp】

    题目链接 洛谷P1912[原题,需输出方案] BZOJ1563[无SPJ,只需输出结果] 题解 四边形不等式 什么是四边形不等式? 一个定义域在整数上的函数\(val(i,j)\),满足对\(\for ...

随机推荐

  1. 使用语句清除sqlserver数据库日志文件

    修改其中的3个参数(数据库名,日志文件名,和目标日志文件的大小),运行即可 SET NOCOUNT ON DECLARE @LogicalFileName sysname, @MaxMinutes I ...

  2. JDK8新特性一览

    转载自:http://blog.csdn.net/qiubabin/article/details/70256683 官方新特性说明地址 Jdk8新特性.png 下面对几个常用的特性做下重点说明. 一 ...

  3. JAVA基础2——类初始化相关执行顺序

    类初始化相关执行顺序 几个概念说明 代码块的含义与作用 static静态代码块: 一般用于初始化类中的静态变量.比如:给静态的数组或者list变量赋初值.使用static静态代码块进行初始化与直接在定 ...

  4. PHP+Redis 实例【二】页面缓存 新玩法

    今天算是认识到博客园里的审查团队多内幕了,哈哈,贴个图玩下. 气死宝宝了. 进入主题! 今天就不写什么功能性的了,换下口味说下关于页面级的缓存,应该怎么做. 相信有很多小伙伴查了百度,甚至google ...

  5. azure上连续部署web

    连续部署web   连续部署web,可以在第一次部署完web应用后,方便修改和自动提交代码部署新版本的web应用.其中自动提交使用github中的webhook,使代码在master上提交修改后可以自 ...

  6. 实用的jQuery技巧

    1.回到顶部按钮  利用jQuery里的animate和scrollTop方法,你便不需要使用插件创建简单的滚动到顶部动画. // Back to top $('.top').click(functi ...

  7. Python中的冒泡排序

    冒泡排序 冒泡排序(英语:Bubble Sort)是一种简单的排序算法.它重复地遍历要排序的数列,一次比较两个元素,如果他们的顺序错误就把他们交换过来.遍历数列的工作是重复地进行直到没有再需要交换,也 ...

  8. Akka(41): Http:DBTable-rows streaming - 数据库表行交换

    在前面一篇讨论里我们介绍了通过http进行文件的交换.因为文件内容是以一堆bytes来表示的,而http消息的数据部分也是byte类型的,所以我们可以直接用Source[ByteString,_]来读 ...

  9. Date( )方法 章节中,你可以查看更多关于日期转换为字符串的函数

    在 Date 方法 章节中,你可以查看更多关于日期转换为字符串的函数: 方法 描述 getDate() 从 Date 对象返回一个月中的某一天 (1 ~ 31). getDay() 从 Date 对象 ...

  10. mybatis实现延迟加载多对一

    1.数据库表 CREATE TABLE `country` ( `cid` ) NOT NULL AUTO_INCREMENT COMMENT '国家id', `cname` ) COLLATE ut ...