BZOJ1563/洛谷P1912 诗人小G 【四边形不等式优化dp】
题目链接
洛谷P1912【原题,需输出方案】
BZOJ1563【无SPJ,只需输出结果】
题解
四边形不等式
什么是四边形不等式?
一个定义域在整数上的函数\(val(i,j)\),满足对\(\forall a \le b \le c \le d\)有
\]
那么我们称函数\(val(i,j)\)满足四边形不等式
一般地,当我们需要证明一个函数\(val(i,j)\)满足四边形不等式时,只需证对于\(\forall j < i\)有
\]
因为若该条件满足,
那么有
对于\(j < i\)有
\]
对于\(j + 1 < i\)有
\]
两式相加
\]
同理,只要满足第一个条件,对于\(\forall j + x < i + y\)都能推出
\]
所以对\(\forall a \le b \le c \le d\)有
\]
证毕
如何使用四边形不等式优化\(dp\)?
如果我们有这样一个递推式
\]
如果是\(1D1D\)方程,自然可以单调队列或者斜率优化
但是如果\(val(i,j)\)比较复杂,无法展开,我们不能有效地分离\(i,j\)变量,斜率优化就失效了
这个时候就要考虑\(val(j,i)\)是否满足四边形不等式
假使\(val(j,i)\)是满足的
那么我们对\(f[i]\)求出了一个最优转移位置,即为\(p[i]\)
那么对于\(\forall j < p[i]\),都有
\]
我们现在要求\(f[k]\),其中\(k > i\)
以上我们有\(j < p[i] < i < k\)
那么由四边形不等式:
\]
交换一下
\]
与\(f[p[i]] + val(p[i],i) \le f[j] + val(j,i)\)相加
得
\]
得证\(f[k]\)的决策\(p[k] \ge p[i]\)
换言之,\(f[i]\)的决策满足决策单调性
对于\(\forall i > j\)都有\(p[i] \ge p[j]\)
我们就可以从这里入手优化这个\(O(n^2)\)的转移
为求出\(f[i]\),我们只需求出\(p[i]\)数组
一开始令\(p[i] = 0\),即未开始转移前所有位置的最优决策为位置\(0\)
之后从小枚举\(i\)
枚举到\(i\)时,\(i\)处最优决策更新完毕,那么\(\forall j \le i\),\(p[j]\)和\(f[j]\)都已经计算
为了更新\(p[i]\)数组,我们只需找到\(i\)所能更新的最左的位置\(l\),由于决策的单调性,\([l,n]\)的决策都将更新为\(i\)
此时二分判断即可
由于需要进行区间赋值,我们可以用一个队列来优化以上操作
具体地,储存若干三元组\((pos,l,r)\),表示区间\([l,r]\)的决策为\(pos\),每次更新从队尾逐个取出检查即可
复杂度优化为\(O(nlogn)\)
诗人小G
设\(f[i]\)为前\(i\)个句子排版的最小不和谐度
记\(len[i]\)为句子\(i\)的长度,\(sum[i]\)为句子长度前缀和
容易写出转移方程
\]
我们记\(val(j,i) = |(sum[i] - sum[j]) + (i - j - 1) - L|^{P}\),显然无法进行斜率优化
考虑\(val(j,i)\)是否满足四边形不等式,即\(f[i]\)的决策是否具有单调性
我们可以打表证明
要证\(val(j,i)\)满足四边形不等式,只需证对\(\forall j < i\)
\]
即证
\]
观察\(val(j,i) = |(sum[i] + i) - (sum[j] + j) - (L + 1)|^{P}\)
我们令\(x = (sum[i] + i) - (sum[j] + j) - (L + 1)\)
我们令\(y = (sum[i] + i) - (sum[j + 1] + j + 1) - (L + 1)\)
那么原式化为:
\]
又因为\(x > y\)
我们只需证对于函数\(f(x) = |x|^{P} - |x + c|^{P}\)关于\(x\)单调递减,其中\(c\)为大于\(0\)的常数
可以使用导数分类讨论得证
于是就可以用四边形不等式优化成\(O(nlogn)\)
不输出方案版:
#include<algorithm>
#include<iostream>
#include<cstring>
#include<cstdio>
#include<cmath>
#include<map>
#define Redge(u) for (int k = h[u],to; k; k = ed[k].nxt)
#define REP(i,n) for (int i = 1; i <= (n); i++)
#define mp(a,b) make_pair<int,int>(a,b)
#define cls(s) memset(s,0,sizeof(s))
#define cp pair<int,int>
#define LL long double
using namespace std;
const int maxn = 100005,maxm = 100005,INF = 1000000000;
inline int read(){
	int out = 0,flag = 1; char c = getchar();
	while (c < 48 || c > 57){if (c == '-') flag = -1; c = getchar();}
	while (c >= 48 && c <= 57){out = (out << 3) + (out << 1) + c - 48; c = getchar();}
	return out * flag;
}
inline void write(LL x){
	if (x / 10 >= 1) write(x / 10);
	putchar((int)floor(x - floor(x / 10) * 10) + '0');
}
int n,P,L;
LL f[maxn],a[maxn],sum[maxn];
char s[maxn][32];
int head,tail;
struct node{
	int j,l,r;
}q[maxn * 3];
LL qpow(LL a,int b){
	LL re = 1;
	for (; b; b >>= 1,a = a * a)
		if (b & 1) re = re * a;
	return re;
}
LL val(int i,int j){
	return qpow(fabs(sum[i] - sum[j] + (i - j - 1) - L),P);
}
bool check(int a,int b,int i){
	return f[a] + val(i,a) <= f[b] + val(i,b);
}
int main(){
	int T = read();
	while (T--){
		n = read(); L = read(); P = read();
		sum[0] = f[0] = 0;
		REP(i,n){
			scanf("%s",s[i] + 1),a[i] = strlen(s[i] + 1);
			sum[i] = sum[i - 1] + a[i];
			f[i] = val(i,0);
		}
		q[head = tail = 0] = (node){0,1,n};
		node u; int l,r,mid;
		for (int i = 1; i <= n; i++){
			f[i] = f[q[head].j] + val(i,q[head].j);
			q[head].l++;
			if (q[head].l > q[head].r) head++;
			while (head <= tail){
				u = q[tail--];
				if (check(i,u.j,u.l)){
					if (head > tail){
						q[++tail] = (node){i,i + 1,n};
						break;
					}
				}
				else if (!check(i,u.j,u.r)){
					q[++tail] = u;
					if (u.r == n) break;
					else{
						q[++tail] = (node){i,u.r + 1,n};
						break;
					}
				}
				else {
					l = u.l; r = u.r;
					while (l < r){
						mid = l + r >> 1;
						if (check(i,u.j,mid)) r = mid;
						else l = mid + 1;
					}
					q[++tail] = (node){u.j,u.l,l - 1};
					q[++tail] = (node){i,l,n};
					break;
				}
			}
		}
		if (f[n] > qpow(10,18)) puts("Too hard to arrange");
		else write(f[n]),puts("");
		printf("--------------------");
		if (T) puts("");
	}
	return 0;
}
输出方案版
#include<algorithm>
#include<iostream>
#include<cstring>
#include<cstdio>
#include<cmath>
#include<map>
#define Redge(u) for (int k = h[u],to; k; k = ed[k].nxt)
#define REP(i,n) for (int i = 1; i <= (n); i++)
#define mp(a,b) make_pair<int,int>(a,b)
#define cls(s) memset(s,0,sizeof(s))
#define cp pair<int,int>
#define LL long double
using namespace std;
const int maxn = 100005,maxm = 100005,INF = 1000000000;
inline int read(){
	int out = 0,flag = 1; char c = getchar();
	while (c < 48 || c > 57){if (c == '-') flag = -1; c = getchar();}
	while (c >= 48 && c <= 57){out = (out << 3) + (out << 1) + c - 48; c = getchar();}
	return out * flag;
}
inline void write(LL x){
	if (x / 10 >= 1) write(x / 10);
	putchar((int)floor(x - floor(x / 10) * 10) + '0');
}
int n,P,L,p[maxn];
LL f[maxn],a[maxn],sum[maxn];
char s[maxn][32];
int head,tail;
struct node{
	int j,l,r;
}q[maxn * 3];
LL qpow(LL a,int b){
	LL re = 1;
	for (; b; b >>= 1,a = a * a)
		if (b & 1) re = re * a;
	return re;
}
LL val(int i,int j){
	return qpow(fabs(sum[i] - sum[j] + (i - j - 1) - L),P);
}
bool check(int a,int b,int i){
	return f[a] + val(i,a) <= f[b] + val(i,b);
}
void print(int u){
	if (p[u]) print(p[u]);
	for (int i = p[u] + 1; i <= u; i++){
		printf("%s",s[i] + 1);
		putchar(i < u ? ' ' : '\n');
	}
}
int main(){
	int T = read();
	while (T--){
		n = read(); L = read(); P = read();
		sum[0] = f[0] = 0;
		REP(i,n){
			scanf("%s",s[i] + 1),a[i] = strlen(s[i] + 1);
			sum[i] = sum[i - 1] + a[i];
			f[i] = val(i,0);
		}
		q[head = tail = 0] = (node){0,1,n};
		node u; int l,r,mid;
		for (int i = 1; i <= n; i++){
			f[i] = f[q[head].j] + val(i,q[head].j);
			p[i] = q[head].j;
			q[head].l++;
			if (q[head].l > q[head].r) head++;
			while (head <= tail){
				u = q[tail--];
				if (check(i,u.j,u.l)){
					if (head > tail){
						q[++tail] = (node){i,i + 1,n};
						break;
					}
				}
				else if (!check(i,u.j,u.r)){
					q[++tail] = u;
					if (u.r == n) break;
					else{
						q[++tail] = (node){i,u.r + 1,n};
						break;
					}
				}
				else {
					l = u.l; r = u.r;
					while (l < r){
						mid = l + r >> 1;
						if (check(i,u.j,mid)) r = mid;
						else l = mid + 1;
					}
					q[++tail] = (node){u.j,u.l,l - 1};
					q[++tail] = (node){i,l,n};
					break;
				}
			}
		}
		if (f[n] > qpow(10,18)) puts("Too hard to arrange");
		else{
			write(f[n]),puts("");
			print(n);
		}
		printf("--------------------");
		if (T) puts("");
	}
	return 0;
}
												
											BZOJ1563/洛谷P1912 诗人小G 【四边形不等式优化dp】的更多相关文章
- [BZOJ1563][NOI2009]诗人小G(决策单调性优化DP)
		
模板题. 每个决策点都有一个作用区间,后来的决策点可能会比先前的优.于是对于每个决策点二分到它会比谁在什么时候更优,得到新的决策点集合与区间. #include<cstdio> #incl ...
 - [NOI2009]诗人小G  决策单调性优化DP
		
第一次写这种二分来优化决策单调性的问题.... 调了好久,,,各种细节问题 显然有DP方程: $f[i]=min(f[j] + qpow(abs(sum[i] - sum[j] - L - 1))); ...
 - 洛谷 P4093 [HEOI2016/TJOI2016]序列 CDQ分治优化DP
		
洛谷 P4093 [HEOI2016/TJOI2016]序列 CDQ分治优化DP 题目描述 佳媛姐姐过生日的时候,她的小伙伴从某宝上买了一个有趣的玩具送给他. 玩具上有一个数列,数列中某些项的值可能会 ...
 - CF321E Ciel and Gondolas Wqs二分 四边形不等式优化dp 决策单调性
		
LINK:CF321E Ciel and Gondolas 很少遇到这么有意思的题目了.虽然很套路.. 容易想到dp \(f_{i,j}\)表示前i段分了j段的最小值 转移需要维护一个\(cost(i ...
 - hdu 2829 Lawrence(四边形不等式优化dp)
		
T. E. Lawrence was a controversial figure during World War I. He was a British officer who served in ...
 - 【转】斜率优化DP和四边形不等式优化DP整理
		
(自己的理解:首先考虑单调队列,不行时考虑斜率,再不行就考虑不等式什么的东西) 当dp的状态转移方程dp[i]的状态i需要从前面(0~i-1)个状态找出最优子决策做转移时 我们常常需要双重循环 (一重 ...
 - codevs3002石子归并3(四边形不等式优化dp)
		
3002 石子归并 3 参考 http://it.dgzx.net/drkt/oszt/zltk/yxlw/dongtai3.htm 时间限制: 1 s 空间限制: 256000 KB 题目等级 ...
 - 洛谷 P3580 - [POI2014]ZAL-Freight(单调队列优化 dp)
		
洛谷题面传送门 考虑一个平凡的 DP:我们设 \(dp_i\) 表示前 \(i\) 辆车一来一回所需的最小时间. 注意到我们每次肯定会让某一段连续的火车一趟过去又一趟回来,故转移可以枚举上一段结束位置 ...
 - 学习笔记:四边形不等式优化 DP
		
定义 & 等价形式 四边形不等式是定义在整数集上的二元函数 \(w(x, y)\). 定义:对于任意 \(a \le b \le c \le d\),满足交叉小于等于包含(即 \(w(a, c ...
 
随机推荐
- Vue 数组封装和组件data定义为函数一些猜测
			
数组封装 var vm={ list:[0,1] } var push=vm.list.push;//把数组原来的方法存起来 vm.list.push=function(arg){//重新定义数组的 ...
 - 原生android(二)——认识activity
			
一.activity的生命周期 1.onCreate():在活动第一次被创建的时候调用,用来完成活动的初始化操作,如加载布局.绑定事件等 2.onStart():在活动由不可见变为可见时被调用 3.o ...
 - qt cout输出中文乱码解决记录
			
工具 -> 选项-> 文本编辑器-> 行为 -> 文件编码->默认编码改为System 乱码原因: 默认用utf-8编码,控制台默认gbk编码,编码不一致导致的乱码
 - SQL Server变量杂谈
			
学习SQL的过程有进步的话还是一件很美妙的事情的 在第一家公司虽然只呆了两年,但是感觉是我进步最快的两年.那时候工作和生活都挺充实的,每天都有一点点的收获和付出,其中最大的收获莫过于掌握一些核心技能. ...
 - vs2017 asp.net FriendlyUrls 新特性
			
这个包如何使用呢?其实很简单,只需将nuget包添加到项目中,再调用routes.EnableFriendlyUrls(),你就可以通过/Foo来访问/Foo.aspx了!你也能够利用URL片段将更多 ...
 - pyqt5实现SMTP邮件发送
			
# -*- coding: utf-8 -*- # Form implementation generated from reading ui file 'SMTP.ui' # # Created b ...
 - Python 中的实用数据挖掘
			
本文是 2014 年 12 月我在布拉格经济大学做的名为‘ Python 数据科学’讲座的笔记.欢迎通过 @RadimRehurek 进行提问和评论. 本次讲座的目的是展示一些关于机器学习的高级概念. ...
 - 带你玩转JavaScript中的隐式强制类型转换
			
正题开始前我想先抛出一个问题,==和===有什么区别?可能一般人会想,不就是后者除了比较值相等之外还会比较类型是否相等嘛,有什么好问的,谁不知道?!但是这样说还不够准确,两者的真正区别其实是==在比较 ...
 - 六周psp
			
本周psp 本周进度条 代码累积折线图 博文字数累积折线图 饼状图
 - 团队backlog和燃尽图
			
首先我们的团队的任务是做一款一对一交流的软件,我们团队的backlog如下: 用电脑搭建一个服务器一对一平台 实现数据库和Android的链接 编写登陆,注册页面,和登陆成功界面,和登陆失败页面 实现 ...