bzoj 2726 [SDOI2012]任务安排(斜率DP+CDQ分治)
【题目链接】
http://www.lydsy.com/JudgeOnline/problem.php?id=2726
【题意】
将n个任务划分成若干个块,每一组Mi任务花费代价(T+sigma{ tj }+s)*sima{ fi },j属于Mi,T为当前时间,问最小代价。
【思路】
设f[i]为将前i个任务划分完成的最小费用,Ti Fi分别表示t和f的前缀和,则不难写出转移方程式:
f[i]=min{ f[j]+(F[n]-F[j])*(T[i]-T[j]+s) },1<=j<=i-1
经过整理得到:
f[i]=min{ -T[i]*F[j]+(f[j]-F[n]*T[j]+F[j]*T[j]-s*F[j]) }
设X(i)=F[i],Y(i)=f[j]-F[n]*T[j]+F[j]*T[j]-s*F[j],a(i)=T[i]则有:
f[i]=min{ -a(i)*X(j)+Y(j) }
则我们需要:
min p = -a(i)*X(j)+Y(j)
即最小化直线方程:Y(j)=a(i)*X(j)+p的截距。
如果时间没有负数的话(出题人神脑洞<_<,这个题可以直接上单调队列维护下凸线。事实上这个算法可以得到60分。
有了负数以后因为斜率不满足随x单调,所以不能使用单调队列。
然后这道题就需要用splay或CDQ分治解。
Splay写法还是不会=-=,CDQ分治的处理和 这道题 类似,就不再赘述了。
需要注意的:x,y可能到达long long级别,所以不要直接除得slop
运算过程中long long的标识。
归并数组的时候几个小条件。
【代码】
#include<set>
#include<cmath>
#include<queue>
#include<vector>
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
#define trav(u,i) for(int i=front[u];i;i=e[i].nxt)
#define FOR(a,b,c) for(int a=(b);a<=(c);a++)
#define rep(a,b,c) for(int a=(b);a>=(c);a--)
using namespace std; typedef long long ll;
const int N = 5e5+;
const ll inf = 1e18; ll read() {
char c=getchar();
ll f=,x=;
while(!isdigit(c)) {
if(c=='-') f=-; c=getchar();
}
while(isdigit(c))
x=x*+c-'',c=getchar();
return x*f;
} struct Pt
{
ll x,y,k;
int id;
bool operator < (const Pt& rhs) const
{
return k<rhs.k;
}
}q[N],t[N]; bool cmp(const Pt& a,const Pt& b)
{
return a.x<b.x || a.x==b.x&&a.y<b.y;
} int T[N],F[N],st[N],top,n,s; ll f[N]; ll U(int j,int k)
{
return (ll)q[k].y-q[j].y;
}
ll D(int j,int k)
{
return (ll)q[k].x-q[j].x;
} void solve(int l,int r)
{
if(l==r) {
q[l].x=(ll)F[l];
q[l].y=(ll)f[l]-(ll)F[n]*T[l]+(ll)F[l]*T[l]-(ll)s*F[l];
return ;
}
int mid=l+r>>;
int l1=l,l2=mid+;
FOR(i,l,r) {
if(q[i].id>mid) t[l2++]=q[i];
else t[l1++]=q[i];
}
memcpy(q+l,t+l,sizeof(Pt)*(r-l+));
solve(l,mid);
top=;
FOR(i,l,mid) {
while(top> && (ll)U(st[top-],st[top])*D(st[top-],i)>(ll)U(st[top-],i)*D(st[top-],st[top])) top--;
st[++top]=i;
}
int j=;
FOR(i,mid+,r) {
while(j<top && U(st[j],st[j+])<(ll)q[i].k*D(st[j],st[j+])) j++;
f[q[i].id]=min(f[q[i].id],-(ll)q[i].k*q[st[j]].x+q[st[j]].y+(ll)F[n]*(T[q[i].id]+s));
}
solve(mid+,r);
l1=l,l2=mid+;
FOR(i,l,r) {
if((l2>r||cmp(q[l1],q[l2]))&&l1<=mid) t[i]=q[l1++];
else t[i]=q[l2++];
}
memcpy(q+l,t+l,sizeof(Pt)*(r-l+));
} int main()
{
// freopen("in.in","r",stdin);
// freopen("out.out","w",stdout);
n=read(),s=read();
FOR(i,,n)
T[i]=read(),T[i]+=T[i-],
F[i]=read(),F[i]+=F[i-];
FOR(i,,n) {
q[i].id=i;
q[i].k=T[i];
f[i]=inf;
}
sort(q+,q+n+);
solve(,n);
printf("%lld\n",f[n]);
return ;
}
简单DP(20分)
#include<set>
#include<cmath>
#include<queue>
#include<vector>
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
#define trav(u,i) for(int i=front[u];i;i=e[i].nxt)
#define FOR(a,b,c) for(int a=(b);a<=(c);a++)
#define rep(a,b,c) for(int a=(b);a>=(c);a--)
using namespace std; typedef long long ll;
const int N = 2e5+;
const ll inf = 1e18+; ll read() {
char c=getchar();
ll f=,x=;
while(!isdigit(c)) {
if(c=='-') f=-; c=getchar();
}
while(isdigit(c))
x=x*+c-'',c=getchar();
return x*f;
} ll T[N],F[N],f[N],pre[N],n,s; int main()
{
freopen("in.in","r",stdin);
freopen("outr.out","w",stdout);
n=read(),s=read();
FOR(i,,n)
T[i]=read(),T[i]+=T[i-],F[i]=read(),F[i]+=F[i-] ;
FOR(i,,n) {
f[i]=inf;
FOR(j,,i-) {
//f[i]=min(f[i],f[j]+(F[n]-F[j])*(T[i]-T[j]+s));
int tmp=-T[i]*F[j]+(f[j]-F[n]*T[j]+F[j]*T[j]-s*F[j]);
if(tmp<f[i]) pre[i]=j,f[i]=tmp;
}
f[i]+=F[n]*(T[i]+s);
}
printf("%lld\n",f[n]);
//FOR(i,1,n) {
// printf("%d %d\n",F[i],f[i]-F[n]*T[i]+F[i]*T[i]-s*F[i]);
//}
return ;
}
bzoj 2726 [SDOI2012]任务安排(斜率DP+CDQ分治)的更多相关文章
- BZOJ 2726: [SDOI2012]任务安排 [斜率优化DP 二分 提前计算代价]
2726: [SDOI2012]任务安排 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 868 Solved: 236[Submit][Status ...
- 【BZOJ2726】[SDOI2012]任务安排 斜率优化+cdq分治
[BZOJ2726][SDOI2012]任务安排 Description 机器上有N个需要处理的任务,它们构成了一个序列.这些任务被标号为1到N,因此序列的排列为1,2,3...N.这N个任务被分成若 ...
- BZOJ 2726 [SDOI2012] 任务安排 - 斜率优化dp
题解 转移方程与我的上一篇题解一样 : $S\times sumC_j + F_j = sumT_i \times sumC_j + F_i - S \times sumC_N$. 分离成:$S\t ...
- BZOJ 2726: [SDOI2012]任务安排 斜率优化 + 凸壳二分 + 卡精
Code: #include<bits/stdc++.h> #define setIO(s) freopen(s".in","r",stdin) # ...
- BZOJ 2726: [SDOI2012]任务安排( dp + cdq分治 )
考虑每批任务对后面任务都有贡献, dp(i) = min( dp(j) + F(i) * (T(i) - T(j) + S) ) (i < j <= N) F, T均为后缀和. 与j有关 ...
- bzoj 2244 [SDOI2011]拦截导弹(DP+CDQ分治+BIT)
[题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=2244 [题意] 给定n个二元组,求出最长不上升子序列和各颗导弹被拦截的概率. [思路] ...
- bzoj 1492 [NOI2007]货币兑换Cash(斜率dp+cdq分治)
[题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=1492 [题意] 有AB两种货币,每天可以可以付IPi元,买到A券和B券,且A:B= ...
- bzoj 2726: [SDOI2012]任务安排
Description 机 器上有N个需要处理的任务,它们构成了一个序列.这些任务被标号为1到N,因此序列的排列为1,2,3...N.这N个任务被分成若干批,每批包含相邻的 若干任务.从时刻0开始,这 ...
- 斜率dp cdq 分治
f[i] = min { f[j] + sqr(a[i] - a[j]) } f[i]= min { -2 * a[i] * a[j] + a[j] * a[j] + f[j] } + a[i] * ...
随机推荐
- 带复杂类的list,list<class>前台往后台传输
1.前台 $("#applyGoods").click(function(){ var usid=$(this).next().text(); var aid=$(this).ne ...
- 微信小程序 功能函数 替换字符串内的指定字符
var str = 'abcadeacf'; var str1 = str.replace('a', 'o'); alert(str1); // 打印结果: obcadeacf var st ...
- CentOS6.7的安装
VMware9的安装请阅读: http://www.cnblogs.com/duanji/p/yueding.html CentOS6.7在VMware9中安装 1.启动VMware的画面 2.点击 ...
- MT【126】点对个数两题之二【图论】
在平面上有\(n\) 个点$S={x_1,x_2\cdots,x_n}, $ 证明在这 \(n\) 个点中距离为 \(1\) 的点对数不超过 \(\dfrac{n}{4}+\dfrac{2}{2}n^ ...
- MySql数据库迁移图文展示
MySql数据库的数据从一台服务器迁移到另外一台服务器需要将数据库导出,再从另外一台服务器导入.方法有很多,MySql配套的相关工具都有这个功能.phpMyAdmin就可以做,但是这个加载起来慢,推荐 ...
- fzyzojP2119 -- 圆圈游戏
说白了,就是这个样子: 这个玩意明显是一个优美的树形结构 是个森林 然后建个虚点0,并且w[0]=0,然后树形dp即可 f[x]=max(w[x],∑f[son]) 难点是:树怎么建? 就要上计算几何 ...
- Luogu P3251 [JLOI2012]时间流逝 期望dp
题面 题面 题解 期望\(dp\)好题! 今年\(ZJOI\)有讲过这题... 首先因为\(T\)只有\(50\),大力\(dfs\)后发现,可能的状态数最多只有\(20w\)左右,所以我们就可以大力 ...
- 微服务Kong(一)——简介
重要提示: 本教程是根据 KONG 0.10.x 版本进行编写的. 一.什么是KONG Kong是一个可扩展的开源API层(也称为API网关或API中间件).它运行在任何RESTful API之前,并 ...
- pg中删除的页是否仍被访问
昨天看到微信群中,有人提问:pg对于标记为删除的页,是否会扫描到? 今天做了一下测试,发现如果删除的是表的最后连续的几个页(根据ctid来确定数据插入先后,只讨论有insert的情况)中的数据,最后几 ...
- HDU--4607
题目: Park Visit 原题链接:http://acm.hdu.edu.cn/showproblem.php?pid=4607 分析:求树的直径.所谓树的直径,指的是一棵树里任意两点之间的最远距 ...