【题目链接】

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分治)的更多相关文章

  1. BZOJ 2726: [SDOI2012]任务安排 [斜率优化DP 二分 提前计算代价]

    2726: [SDOI2012]任务安排 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 868  Solved: 236[Submit][Status ...

  2. 【BZOJ2726】[SDOI2012]任务安排 斜率优化+cdq分治

    [BZOJ2726][SDOI2012]任务安排 Description 机器上有N个需要处理的任务,它们构成了一个序列.这些任务被标号为1到N,因此序列的排列为1,2,3...N.这N个任务被分成若 ...

  3. BZOJ 2726 [SDOI2012] 任务安排 - 斜率优化dp

    题解 转移方程与我的上一篇题解一样 : $S\times sumC_j  + F_j = sumT_i \times sumC_j + F_i - S \times sumC_N$. 分离成:$S\t ...

  4. BZOJ 2726: [SDOI2012]任务安排 斜率优化 + 凸壳二分 + 卡精

    Code: #include<bits/stdc++.h> #define setIO(s) freopen(s".in","r",stdin) # ...

  5. BZOJ 2726: [SDOI2012]任务安排( dp + cdq分治 )

    考虑每批任务对后面任务都有贡献, dp(i) = min( dp(j) + F(i) * (T(i) - T(j) + S) ) (i < j <= N)  F, T均为后缀和. 与j有关 ...

  6. bzoj 2244 [SDOI2011]拦截导弹(DP+CDQ分治+BIT)

    [题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=2244 [题意] 给定n个二元组,求出最长不上升子序列和各颗导弹被拦截的概率. [思路] ...

  7. bzoj 1492 [NOI2007]货币兑换Cash(斜率dp+cdq分治)

    [题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=1492   [题意] 有AB两种货币,每天可以可以付IPi元,买到A券和B券,且A:B= ...

  8. bzoj 2726: [SDOI2012]任务安排

    Description 机 器上有N个需要处理的任务,它们构成了一个序列.这些任务被标号为1到N,因此序列的排列为1,2,3...N.这N个任务被分成若干批,每批包含相邻的 若干任务.从时刻0开始,这 ...

  9. 斜率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] * ...

随机推荐

  1. 6/11 sprint2 看板和燃尽图的更新

  2. Js获取上一月份

    new Date(new Date().setMonth(new Date().getMonth() - 1))

  3. Linux下CPU信息的查看

    逻辑CPU个数: cat /proc/cpuinfo | grep "processor" | wc -l     //32 物理CPU个数: cat /proc/cpuinfo ...

  4. RFC 8446

    https://tools.ietf.org/html/rfc8446#section-2.3 简要内容.. [Docs] [txt|pdf] [draft-ietf-tls-...] [Tracke ...

  5. SQL语句查询一个数据库中的所有表

    --读取库中的所有表名 select name from sysobjects where xtype='u' --读取指定表的所有列名 select name from syscolumns whe ...

  6. jmeter 正则表达式提取

    引用名称:自己定义的变量名称,后续请求将要引用到的变量名,如填写的是:id,后面的引用方式是${id} 正则表达式:提取内容的正则表达式,相当于lr中的关联函数 [()     括起来的部分就是需要提 ...

  7. centOS基本操作和命令(更新)

    1.文字输入和图形界面切换 CTRL+ALT+(F1~F6)为切换至文字输入,分别对应六个不同输入界面,可用以不同账号:CTRL+ALT+F7为切换至图形界面 2.修改时区 date -R date ...

  8. sqlserver创建同义词

    https://blog.csdn.net/anderslu/article/details/53433841?utm_source=itdadao&utm_medium=referral 例 ...

  9. BZOJ5118 Fib数列2(矩阵快速幂)

    特殊矩阵的幂同样满足费马小定理. #include<iostream> #include<cstdio> #include<cmath> #include<c ...

  10. Statement和PreparedStatement之间的区别

    Statement和PreparedStatement之间的区别: 1.PreparedStatement是预编译的,对于批量处理可以大大提高效率. 也叫JDBC存储过程2.使用 Statement ...