正题

题目链接:https://www.luogu.com.cn/problem/P1912


题目大意

给出\(n\)个字符串,把这些字符串依次用空格(算一个长度)连接分成若干段,若一段长度为\(x\),那么代价是\(|x-L|^P\)

求代价和最小的方案,如果代价大于\(1e18\)则输出其他东西

\(1\leq n\leq 10^5,1\leq L\leq 3\times 10^6,1\leq P\leq 10\)


解题思路

\(s_i\)表示前\(i\)个字符串的长度和加\(i\),那么有转移方程

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

这个转移很麻烦不能直接用单调队列之类的优化,但是它满足四边形不等式

\(w_{i,j}=|s_i-s_j-1-L|^P\),然后满足

\[w_{i,j}+w_{i+1,j+1}\leq w_{i,j+1}+w_{i+1,j}
\]

这里就不证明了,因为证明需要用到求导。

感谢理解的话可以发现因为有个\(abs\),所以对于一个决策来说是先下后上,而且两个决策最多只有一个交点。

所以有决策单调性,我们用单调队列维护一个该决策和它的下一个决策的交换点\(k_i\),然后每次判断新加入的点与队尾的前一个的交换点是否会代替掉队尾即可。

求交换点的话用二分就好了。

时间复杂度\(O(Tn\log n)\)

怕转移太大可以用\(long\ double\)存,因为如果很大的时候精度就不需要管了,我们只需要知道它是否超过\(1e18\)就好了。


code

#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cmath>
#define ll long double
using namespace std;
const int N=1e5+10;
int T,n,L,P,p[N],k[N],q[N];
ll f[N],s[N];
char st[N][31];
ll power(ll x,int b){
ll ans=1;
while(b){
if(b&1)ans=ans*x;
x=x*x;b>>=1;
}
return ans;
}
ll calc(int j,int i)
{return f[j]+power(fabs(s[i]-s[j]-1-L),P);}
int bound(int i,int j){
int l=i,r=n;
while(l<=r){
int mid=(l+r)>>1;
if(calc(i,mid)<calc(j,mid))l=mid+1;
else r=mid-1;
}
return l;
}
void print(int n){
if(!n)return;print(p[n]);
for(int i=p[n]+1;i<n;i++)
printf("%s ",st[i]);
puts(st[n]);
}
int main()
{
scanf("%d",&T);
while(T--){
scanf("%d%d%d",&n,&L,&P);
for(int i=1;i<=n;i++){
scanf("%s",st[i]);
s[i]=s[i-1]+strlen(st[i])+1;
}
int head=1,tail=1;q[1]=0;
for(int i=1;i<=n;i++){
while(head<tail&&k[head]<=i)head++;
f[i]=calc(q[head],i);p[i]=q[head];
while(head<tail&&k[tail-1]>=bound(q[tail],i))tail--;
k[tail]=bound(q[tail],i);q[++tail]=i;
}
if(f[n]>1e18)puts("Too hard to arrange");
else printf("%lld\n",(long long)f[n]),print(n);
puts("--------------------");
}
return 0;
}

P1912-[NOI2009]诗人小G【四边形不等式,单调队列】的更多相关文章

  1. P1912 [NOI2009]诗人小G

    P1912 [NOI2009]诗人小G 思路: 平行四边形不等式优化dp 因为f(j, i) = abs(sum[i]-sum[j]+i-j-1-l)^p 满足平行四边形不等式 j < i f( ...

  2. 不失一般性和快捷性地判定决策单调(洛谷P1912 [NOI2009]诗人小G)(动态规划,决策单调性,单调队列)

    洛谷题目传送门 闲话 看完洛谷larryzhong巨佬的题解,蒟蒻一脸懵逼 如果哪年NOI(放心我这样的蒟蒻是去不了的)又来个决策单调性优化DP,那蒟蒻是不是会看都看不出来直接爆\(0\)?! 还是要 ...

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

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

  4. [NOI2009]诗人小G 四边形优化DP

    题目传送门 f[i] = min(f[j] + val(i,j); 其中val(i,j) 满足 四边形dp策略. 代码: #include<bits/stdc++.h> using nam ...

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

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

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

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

  7. 1563: [NOI2009]诗人小G

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

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

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

  9. LG1912 [NOI2009]诗人小G

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

  10. [NOI2009] 诗人小G [题解]

    诗人小G 题目大意 给出 \(n\) 个长度不超过 \(30\) 的句子,要求你对其进行排版. 对于每一行,有一个规定的行标准长度 \(L\) ,每一行的不协调度等于该行的实际长度与行标准长度差的绝对 ...

随机推荐

  1. vue中常用插件(货币、日期)

    货币插件: 价格格式化 // https://github.com/vuejs/vuex/blob/dev/examples/shopping-cart/currency.js const digit ...

  2. spring-cloud-sleuth+zipkin追踪服务

    1, 父Maven pom 文件 <parent> <groupId>org.springframework.boot</groupId> <artifact ...

  3. mysql 常用见的错误处理

    一, 创建用户: 命令:CREATE USER 'username'@'host' IDENTIFIED BY 'password'; 说明:username - 你将创建的用户名, host - 指 ...

  4. Tcp三次握手中细节

    TCP/IP协议中,TCP协议提供可靠的连接服务,采用三次握手建立一个连接,如下图所示.主机A为客户机,主机B为服务器 说明:(1)第一次握手:建立连接时,客户端A发送SYN包(SYN=j)到服务器B ...

  5. springboot分页插件的使用

    在springboot工程下的pom.xml中添加依赖 <!--分页 pagehelper --> <dependency> <groupId>com.github ...

  6. 【小技巧】java的List分页

    今天,工作上,由于业务的一些特殊性,需要拿到数据后在java代码中进行分页. 写了一个工具类,记录如下: import java.util.ArrayList; import java.util.Li ...

  7. [SWMM]汇水区特征宽度的计算方法

    SWMM模型产流计算中,有一个比较重要的参数就是子汇水区的特征宽度(width),这个参数对地表汇流时间和峰值有一定的影响.子汇水区特征宽度的计算方法有很多,这里介绍比较常用的两种: (1)采用面积除 ...

  8. JAVA中的策略模式strategy

    原文出自:http://ttitfly.iteye.com/blog/136467 1. 以一个算术运算为例,传统做法为: java 代码 package org.common; public cla ...

  9. reids rdb与aof

    rdb:时合高并发场景,容易备份恢复,会丢失部分数据 1.默认开启的方式,可以进过压缩,可以根据时间点生成快照 2.数据量大的情况下恢复快 3.bgsave一边开启fork保存文件,一边继续响应客户端 ...

  10. MySQL——字符串类型——char(n) 和 varchar(n)

    MySQL 的 char(n) 和 varchar(n) 括号中 n 代表字符的个数,而非字节个数,这里说的字符不论文字种类,假设一个字段的数据类型被规定为 char(2),则可以在这个字段上插入 ' ...