洛谷题目传送门

闲话

看完洛谷larryzhong巨佬的题解,蒟蒻一脸懵逼

如果哪年NOI(放心我这样的蒟蒻是去不了的)又来个决策单调性优化DP,那蒟蒻是不是会看都看不出来直接爆\(0\)?!

还是要想点办法,不失一般性也能快捷地判定决策单调。

对于判定决策单调的分析

再补一句决策单调性的概念:状态转移方程形如\(f_i=\min/\max_{j=1}^{i-1} g_j+w_{i,j}\),且记\(f_i\)的最优决策点为\(p_i\)(也就是\(f_i\)从\(g_{p_i}+w_{i,p_i}\)处转移最优)若满足\(p_i\le p_{i+1}\),则该方程满足决策单调性。(摘自蒟蒻的DP优化总结

显然每个决策\(j\)可以用一个关于\(i\)的函数\(f_j(i)\)表示。

函数的一个重要思想:数形结合!

光靠脑子想不到规律,只好先举一些用语言难以描述的反例。我们的函数不能这样

看到这里Dalao们有没有一点想法呢?蒟蒻反正想到了一点——两个函数必须只有一个交点!在这一点之前一个函数更优,而之后就被永远取代了。

感觉满足条件的函数其实很少,分类讨论一下(如有误欢迎Dalao指教)

直线

显然上面的基本要求都满足。不过要是函数是直线的话都可以用斜率优化搞了(\(k_1x+b_1\ge k_2x+b_2,x\ge\frac{b_2-b_1}{k_1-k_2}\))。

不是直线

为了避免图1的尴尬情况,可能需要所有决策函数之间可以通过平移相互变换(形如\(f_j(x)=f_k(x-a)+b\))。

为了避免图2的尴尬情况,可能需要函数的导函数在各自的定义域内单调递增/递减(注意是导函数不是原函数)。

接着,根据蒟蒻肝过的几个题,好像还有一条规律——

如果导函数递增、求最大值(柠檬),或者导函数递减、求最小值,要用单调栈。

如果导函数递增、求最小值(本题),或者导函数递减、求最大值(Lightning Conductor),要用单调队列。

复杂的函数

蒟蒻见过这一道(Yet another minimization problem

感觉可以看成对于每一种数都有一个函数\(\frac{(c_i-c_j)(c_i-c_j+1)}{2}\),单看这一个是满足决策单调性的(\(c_i\ge c_j\),定义域内的导函数是递增的)。

那么总函数就可以写成\(f_j(i)=g_j+\sum\frac{(c_i-c_j)(c_i-c_j+1)}{2}\),怎么看也不像是不满足决策单调性的。

本题的思路

那么就可以回归本题了。

设\(len_i\)为第\(i\)句的长度,\(s_i=i+\sum\limits_{j=1}^i len_j\)(加上\(i\)是默认一句话后面有空格)

设\(f_i\)为选前\(i\)句的最小代价,我们枚举当前这一行填入最后面的多少个句子,注意行末没有空格,长度要\(-1\),那么有方程

\[f_i=\min\limits_{j=0}^{i-1}\{f_j+|s_i-s_j-1-L|^P\}
\]

容易发现后面这一坨决策函数是关于直线\(x=s_j+1+L\)对称的。把它去绝对值,变成两段,显然左边一段和右边一段的导函数都是递增的,左边恒\(<0\),右边恒\(>0\)。又因为这函数是连续的,所以当然整个函数的导函数也单调递增咯!

用队列维护决策二分栈的过程不再赘述,总结里也有。时间复杂度\(O(Tn\log n)\)

看到Dalao们都记录了一个三元组,可蒟蒻还是觉得没啥必要啊。。。只要保存队列中相邻两个元素的临界值\(k\)就好了吧。

一个写法技巧:

二分决策\(x,y(x<y)\)的临界值的时候,左端点设成\(x\)就好了,没必要设成\(1\)(难怪蒟蒻之前写Lightning Conductor跑得有点慢)

三个坑点:

不管是转移还是输出,都要去掉行末的空格(怪蒟蒻看题不清)

当答案大于\(10^{18}\)的时候开longlong也炸了,所以要用实数以牺牲精度的代价换来更大的值域。然而double真的WA了。于是要开long double。

cmath的pow太慢了容易TLE,要手写快速幂。

#include<cstdio>
#include<cmath>
#include<cstring>
#define RG register
#define R RG int
#define G c=getchar()
#define Calc(i,j) f[j]+qpow(abs(s[i]-s[j]-L))//计算函数值
using namespace std;
typedef long double LD;//开long double
const int N=1e5+9;
int n,L,P,s[N],q[N],k[N],pr[N];
LD f[N];
char str[N][33];
inline int in(){
RG char G;
while(c<'-')G;
R x=c&15;G;
while(c>'-')x*=10,x+=c&15,G;
return x;
}
inline LD qpow(RG LD b){//自己写快速幂
RG LD a=1;
for(R k=P;k;k>>=1,b*=b)
if(k&1)a*=b;
return a;
}
inline int bound(R x,R y){//二分临界值
R l=x,r=n+1,m;//左端点设为x减小常数
while(l<r){
m=(l+r)>>1;
Calc(m,x)>=Calc(m,y)?r=m:l=m+1;
}
return l;
}
int main(){
R T=in(),i,h,t;
while(T--){
n=in();L=in()+1;P=in();//把L处理了一下
for(i=1;i<=n;++i){
if(scanf("%s",str[i]));
s[i]=s[i-1]+strlen(str[i])+1;//记前缀和
}
for(q[i=h=t=1]=0;i<=n;++i){
while(h<t&&k[h]<=i)++h;
f[i]=Calc(i,q[h]);pr[i]=q[h];//记录转移位置方便输出方案
while(h<t&&k[t-1]>=bound(q[t],i))--t;
k[t]=bound(q[t],i);q[++t]=i;
}
if(f[n]>1e18)puts("Too hard to arrange");
else{
printf("%.0Lf\n",f[n]);
for(q[t=0]=i=n;i;q[++t]=i=pr[i]);
for(;t;--t){
for(i=q[t]+1;i<q[t-1];++i)
printf("%s ",str[i]);
puts(str[i]);//行末不要搞空格
}
}
puts("--------------------");
}
return 0;
}

不失一般性和快捷性地判定决策单调(洛谷P1912 [NOI2009]诗人小G)(动态规划,决策单调性,单调队列)的更多相关文章

  1. 洛谷P1912 [NOI2009]诗人小G(决策单调性)

    传送门 题解 决策单调性是个啥……导函数是个啥……这题解讲的是啥……我是个啥…… //minamoto #include<iostream> #include<cstdio> ...

  2. [NOI2009]诗人小G --- DP + 决策单调性

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

  3. 2018.09.28 bzoj1563: [NOI2009]诗人小G(决策单调性优化dp)

    传送门 决策单调性优化dp板子题. 感觉队列的写法比栈好写. 所谓决策单调性优化就是每次状态转移的决策都是在向前单调递增的. 所以我们用一个记录三元组(l,r,id)(l,r,id)(l,r,id)的 ...

  4. BZOJ1563 NOI2009 诗人小G【决策单调性优化DP】

    LINK 因为是图片题就懒得挂了 简要题意:有n个串,拼接两个串需要加一个空格,给你l和p,问你拼接后每个串的总长减l的绝对值的p次方的最小值 首先打表发现一下这题是决策单调的对于所有数据都成立就当他 ...

  5. P1912 [NOI2009]诗人小G[决策单调性优化]

    地址 n个数划分若干段,给定$L$,$p$,每段代价为$|sum_i-sum_j-1-L|^p$,求总代价最小. 正常的dp决策单调性优化题目.不知道为什么luogu给了个黑题难度.$f[i]$表示最 ...

  6. [bzoj1563][NOI2009]诗人小G(决策单调性优化)

    题目:http://www.lydsy.com:808/JudgeOnline/problem.php?id=1563 分析: 首先可得朴素的方程:f[i]=min{f[j]+|s[j]-j-s[i] ...

  7. [BZOJ1563][NOI2009]诗人小G(决策单调性优化DP)

    模板题. 每个决策点都有一个作用区间,后来的决策点可能会比先前的优.于是对于每个决策点二分到它会比谁在什么时候更优,得到新的决策点集合与区间. #include<cstdio> #incl ...

  8. BZOJ1563: [NOI2009]诗人小G(决策单调性 前缀和 dp)

    题意 题目链接 Sol 很显然的一个dp方程 \(f_i = min(f_j + (sum_i - sum_j - 1 - L)^P)\) 其中\(sum_i = \sum_{j = 1}^i len ...

  9. P1912-[NOI2009]诗人小G【四边形不等式,单调队列】

    正题 题目链接:https://www.luogu.com.cn/problem/P1912 题目大意 给出\(n\)个字符串,把这些字符串依次用空格(算一个长度)连接分成若干段,若一段长度为\(x\ ...

随机推荐

  1. wpf项目打开多个窗体在任务栏只有一个任务

    原文:wpf项目打开多个窗体在任务栏只有一个任务 如果在wpf里,在一个父窗体上打开子窗体,只在任务栏显示一个任务,不是qq聊天窗口俩人聊天人显示俩给那样,只能显示 一个 private void B ...

  2. 大话设计模式:代理模式 C#

    学无止境,精益求精 十年河东,十年河西,莫欺少年穷 学历代表你的过去,能力代表你的现在,学习代表你的将来 所谓代理模式就是你去委托一个人帮你干一件事!例如:你委托我帮你谈恋爱,你委托我帮你陪你媳妇儿逛 ...

  3. Linux中执行脚本参数获取

    Linux中变量$[#,@,0,1,2,*,$,?]含义 $# 是传给脚本的参数个数 $0 是脚本本身的名字 $1 是传递给该shell脚本的第一个参数 $2 是传递给该shell脚本的第二个参数 $ ...

  4. linux监控文件夹内的文件数量

    开发的时候遇到一个问题,服务器一旦重启,项目生成的文件就丢失了,感觉很莫名其妙..一开始猜测是文件流没有关闭,检查了代码,感觉没毛病.于是先看看是关机丢失了文件还是开机被删除了.下面的脚本每秒执行一次 ...

  5. CSS 内边距 (padding) 实例

    CSS 内边距 (padding) 实例元素的内边距在边框和内容区之间.控制该区域最简单的属性是 padding 属性. CSS padding 属性定义元素边框与元素内容之间的空白区域.CSS 内边 ...

  6. [个人博客作业Week7]软件工程团队项目感想与反思

    在阅读了推荐阅读的材料之后,我想了很多东西.最终还是决定,以团队项目的经历为主线,叙述我关于软件工程的一些思考与体会. 凤凰涅槃,浴火重生 如果要我来概况这几周团队项目的经历的话,那么句话是我所能想到 ...

  7. 【个人博客作业II】代码复审结果

    [代码复审结果] General Does the code work? Does it perform its intended function, the logic is correct etc ...

  8. 四则运算法则在Java中的实现

    软件工程的课程已经上过有一段时间了,前段时间由于比较忙着考试,所以关于四则运算的代码一直没有实现.同时由于近来一段时间一直在自学java,因为C++虽然也是面向对象,而且可以开发很多软件或者程序,但是 ...

  9. github更新,发布地址,燃尽图,总结

    github地址:https://github.com/Lingchaoyang 网盘发布地址:http://pan.baidu.com/s/1qXgHiyC 燃尽图: 团队得分(100分制): 杨灵 ...

  10. Cron任务调度CronNET

    Cron任务调度CronNET 阅读目录 1.Cron介绍和工具 2.CronNET介绍和使用 3.cron-expression-descriptor使用 4.资源 如果用知乎,可以关注专栏:.NE ...