●BZOJ 2149 拆迁队

时间:2021-12-11 03:58:12

题链:

http://www.lydsy.com/JudgeOnline/problem.php?id=2149

题解:

斜率优化DP,栈维护凸包,LIS,分治(我也不晓得是不是CDQ分治...)


一)、解决“旧房子保留”最多
这是一个经典的问题(套路)。
d[i]=a[i]-i,对 d 数组求LIS得到 f[i] 即可。
f[i] 的含义:
1).d 数组中以i位置结尾的最长上升子序列长度;
2).以i位置结尾,且保留i号房子时最多可以保留的房子数。


二)、解决“总花费”最小

g[i] 为 以i结尾,保留i号房子且保留的房子最多时的最小花费

看看如何转移 g 数组:

${g[i]}={min(g[j]}+\frac{(a[j]+1+a[j]+i-j-1)*(i-j-1)}{2}{)}+a[i]+b[i]\;\;{(需满足f[j]+1=f[i])}$

这个DP转移的意思还是很显然的:(以j结尾的贡献+(j+1~i-1)的贡献+i 位置的贡献)

因为j位置和i位置要保留,那么中间的区域$[ j+1, i-1]$要修改,

那自然是贪心地依次定为${a[j]+1,a[j]+2,}\cdots{,a[j]+i-j-1}$的美观值

这样的话,中间的贡献就直接用等差数列求和计算。

然后可以把转移式化成如下形式(...还是很好化的):

${g[i]}={min(g[j]}-{(j+1)a[j]}+\frac{j(j+1)}{2}+{i\cdot d[j]}+\frac{i(i-1)}{2}{)}+a[i]+b[i]$

把取小项里面只含i的项提出来:

$\mathbf{{g[i]}={min(g[j]}-{(j+1)a[j]}+\frac{j(j+1)}{2}+{i\cdot d[j]}{)}+a[i]+b[i]+\frac{i(i-1)}{2}}$

现在这个转移是不是一个典型的可以斜率优化的形式?

(即${g[i]}={只有j为变量的函数}+{i,j为变量且各自的次数都为1的函数}+{只有i为变量的函数}$ )

先令 $y[j]={g[j]}-{(j+1)a[j]}+\frac{j(j+1)}{2}$


错误的尝试:

然后按照斜率优化的套路走,如果对于当前计算的g[i],如果有两个来源 k,j,且k<j,假设j更优。

即 $y[j]+id[j]+a[i]+b[i]+\frac{i(i-1)}{2} < y[k]+id[k]+a[i]+b[i]+\frac{i(i-1)}{2}$

移项后得到$y[j]-y[k] < i(d[k]-d[j])$

接着把不等式的右边的$(d[k]-d[j])$除到左边去么?

可惜虽然我们令 k<j,但是无法保证d[k]和d[j]的大小关系,即d[i]不是单调变化的。

所以不好直接除过去,因为谁知道什么时候会让不等式变号什么时候又不变号呢?


正确的定义:

所以我们改一改上面的定义:

如果对于当前计算的g[i],如果有两个来源 k,j,且d[k]<d[j],假设j更优。

即 $y[j]+id[j]+a[i]+b[i]+\frac{i(i-1)}{2} < y[k]+id[k]+a[i]+b[i]+\frac{i(i-1)}{2}$

移项后得到$y[j]-y[k] < i(d[k]-d[j])$

这时可以保证$(d[k]-d[j])<0$,所以把$(d[k]-d[j])$除过去,

得到$\frac{y[j]-y[k]}{-d[j]-(-d[k])}>i$

令$\mathbf{slope(j,k)=\frac{y[j]-y[k]}{-d[j]-(-d[k])}}$,

那么得到如下结论:如果d[k]<d[j]且slope(j,k)>i,则 j 比 k 的转移优。

然后可以发现,如果存在三个转移来源k,j,i,且d[k]<d[j]<[i],

同时又满足slope(i,j)>slope(j,k),则无论如何j都不会贡献答案,所以可以排除掉j。

这时,没有了d[k]<d[j]<[i],(令x[i]=-d[i],即没有x[i]<x[j]<x[k])且slope(i,j)>slope(j,k)的情况,

如果把 -d[ ]看出x轴,y[ ]看出y轴,那么图像上就只剩下了下凸图形:

●BZOJ 2149 拆迁队


现在有了转移的斜率优化,可是任然有一些问题:

1).斜率优化是建立在d[k]<d[j]的基础上而不是k<j,但是dp转移却要满足j < i且d[j]<=d[i],那如何枚举进行凸包维护和DP转移。

2).转移还有一个强限制 f[i]=f[j]+1,这个又怎么办呢?

做法如下:分治+栈维护+二分,复杂度 $O(Nlog_2^2N)$

首先把所有房子按f值分组,每次用 f=p 的组去贡献 f=p+1 的组(来满足强限制)。

枚举一个p,我们把f值等于p和p+1的位置放在一个数组h里,然后按编号排序。

接下来对h数组进行分治,用f[j]=p的g[j]去得到f[i]=p+1的g[i]。

对于分治的每一层

把 l~mid 里面的f[j]=p 的转移来源提出来到一个数组L,

把 mid+1~r 里面的f[i]=p+1 的位置提出来到另一个数组R,

那么显然用L里面的位置转移到R里面的位置是可以满足dp的转移顺序:从前转移到后面。

然后对L,R数组分别按d值从小到大排序,

对于每一个要计算的g[R[i]],

我们先把L数组里满足d[L[j]]<d[R[i]]的j位置用栈维护好一个下凸包,(由于d[L[j]]随j单调,所以可以直接O(1)往凸包中插入一个点)

显然栈里面维护的凸包的斜率单调递增(下凸包啦),

所以直接在栈里面二分查找到最优的位置作为g[R[i]]的转移来源点即可。


差不多就这样子了, 代码的实现——特别是分治+栈维护这一部分,如果思路还不是特别清楚的话,强烈建议看看代码。

我的错点:

1).维护凸包时,单调栈的栈顶弹出操作需要满足栈里面至少有2个元素才能进行。
2).二分取答案的时候没有写 mid=(l+r)/2,......样例居然还过了???
3).二分取答案写成了一个函数,同时定义了 static int l=1,r=top-1;
导致每次进入函数时没有给 l,r赋初值,......样例居然还是过了???

代码:

#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
#define MAXN 100050
#define ll long long
#define INFi 0x3f3f3f3f
#define INFl 0x3f3f3f3f3f3f3f3fll
using namespace std;
int a[MAXN],b[MAXN],d[MAXN],f[MAXN],h[MAXN][2],cnt;
ll g[MAXN],y[MAXN],ANS2;
int N,ANS1;
struct Link{
int head[MAXN],nxt[MAXN];
Link(){memset(head,-1,sizeof(head));}
void Add(int u,int p){nxt[p]=head[u]; head[u]=p;}
}lk;
struct BIT{
int val[MAXN],N;
int Lowbit(int x){return x&-x;}
void Reset(int n){N=n; memset(val,0xc0,sizeof(val));}
void Modify(int p,int v){while(p<=N) val[p]=max(val[p],v),p=p+Lowbit(p);}
int Query(int p){
static int ret; ret=-INFi;
while(p) ret=max(ret,val[p]),p-=Lowbit(p);
return ret;
}
}DT;
double slope(int i,int j){
return 1.0*(y[i]-y[j])/(-d[i]-(-d[j]));
}
struct STK{
int s[MAXN],top;
void Reset(){top=0;}
void Push(int i){
if(top&&d[i]==d[s[top]]) top--;
while(top>1&&slope(i,s[top])>slope(s[top],s[top-1])) top--;
s[++top]=i;
}
ll Query(int i){
static int l,r,mid,ret;
if(!top) return INFl;
l=1; r=top-1; ret=top;
while(l<=r){
mid=(l+r)>>1;
if(slope(s[mid],s[mid+1])<i) ret=mid,r=mid-1;
else l=mid+1;
}
return y[s[ret]]+1ll*i*d[s[ret]];
}
}S;
void read(int &x){
static int sign; static char ch;
x=0; sign=1; ch=getchar();
while(ch<'0'||'9'<ch){if(ch=='-')sign=-1;ch=getchar();}
while('0'<=ch&&ch<='9'){x=x*10+ch-'0';ch=getchar();}
x=x*sign;
}
bool cmp(int i,int j){
return d[i]==d[j]?y[i]>y[j]:d[i]<d[j];
}
void solve(int l,int r){
static int L[MAXN],R[MAXN],cl,cr;
if(l==r) return;
int mid=(l+r)>>1;
solve(l,mid); solve(mid+1,r); S.Reset(); cl=cr=0;
for(int i=l;i<=mid;i++) if(!h[i][1]) L[++cl]=h[i][0];
for(int i=mid+1;i<=r;i++) if(h[i][1]) R[++cr]=h[i][0];
if(!cl||!cr) return;
sort(L+1,L+cl+1,cmp); sort(R+1,R+cr+1,cmp);
for(int i=1,j=1;i<=cr;i++){
while(j<=cl&&d[L[j]]<=d[R[i]]) S.Push(L[j]),j++;
g[R[i]]=min(g[R[i]],S.Query(R[i]));
}
}
int main(){
static int tmp[MAXN],tnt;
read(N); tmp[tnt=1]=0; ANS2=INFl;
for(int i=1;i<=N;i++) read(a[i]),tmp[++tnt]=d[i]=a[i]-i;
for(int i=1;i<=N;i++) read(b[i]);
sort(tmp+1,tmp+tnt+1);
tnt=unique(tmp+1,tmp+tnt+1)-tmp-1;
DT.Reset(tnt); int p=lower_bound(tmp+1,tmp+tnt+1,0)-tmp;
DT.Modify(p,0);
for(int i=1;i<=N;i++){
g[i]=INFl;
p=lower_bound(tmp+1,tmp+tnt+1,d[i])-tmp;
f[i]=DT.Query(p)+1; DT.Modify(p,f[i]);
}
ANS1=DT.Query(tnt);
for(int i=N;~i;i--) if(f[i]>=0) lk.Add(f[i],i);
for(int q=0,j,k;q<ANS1;q++){
cnt=0; j=lk.head[q]; k=lk.head[q+1];
while(~j||~k){
++cnt;
if(!~j) h[cnt][0]=k,h[cnt][1]=1,k=lk.nxt[k];
else if(!~k||j<k) h[cnt][0]=j,h[cnt][1]=0,j=lk.nxt[j];
else h[cnt][0]=k,h[cnt][1]=1,k=lk.nxt[k];
}//按序号排好序
solve(1,cnt);//分治
for(int i=lk.head[q+1];~i;i=lk.nxt[i]){
g[i]+=1ll*(i-1)*i/2+a[i]+b[i];
y[i]=g[i]-1ll*(i+1)*a[i]+1ll*i*(i+1)/2;
}
}
for(int i=0;i<=N;i++) if(f[i]==ANS1)
ANS2=min(ANS2,g[i]+1ll*(2*a[i]+N-i+1)*(N-i)/2);
printf("%d %lld",ANS1,ANS2);
return 0;
}

  

●BZOJ 2149 拆迁队的更多相关文章

  1. bzoj2149拆迁队 斜率优化dp&plus;分治

    2149: 拆迁队 Time Limit: 10 Sec  Memory Limit: 259 MBSubmit: 397  Solved: 177[Submit][Status][Discuss] ...

  2. BZOJ 3339 &amp&semi; 莫队&plus;&quot&semi;所谓的暴力&quot&semi;

    题意: 给一段数字序列,求一段区间内未出现的最小自然数. SOL: 框架显然用莫队.因为它兹瓷离线. 然而在统计上我打了线段树...用&维护的结点...400w的线段树...然后二分查找... ...

  3. bzoj 2038 莫队算法

    莫队算法,具体的可以看10年莫涛的论文. 大题思路就是假设对于区间l,r我们有了一个答案,那么对于区间l,r+1,我们 可以暴力的转移一个答案,那么对于区间l1,r1和区间l2,r2,需要暴力处理 的 ...

  4. Bzoj2149拆迁队&colon;cdq分治 凸包

    国际惯例的题面:我们考虑大力DP.首先重新定义代价为:1e13*选择数量-(总高度+总补偿).这样我们只需要一个long long就能维护.然后重新定义高度为heighti - i,这样我们能选择高度 ...

  5. bzoj 3289 莫队 逆序对

    莫队维护逆序对,区间左右增减要分类讨论. 记得离散化. /************************************************************** Problem: ...

  6. bzoj 3809 莫队

    收获: 1.分块时顺便记录每个位置所属的块,然后一次排序就OK了. 2.要权衡在“区间移动”与“查询结果”之间的时间,莫队算法一般区间移动频率远大于查询结果,所以我们选择的辅助数据结构时就要注意了,我 ...

  7. bzoj 2038 莫队入门

    http://www.lydsy.com/JudgeOnline/problem.php?id=2038 题意:多次询问区间内取出两个相同颜色的种类数 思路:由于不是在线更新,那么可以进行离线查询,而 ...

  8. bzoj 3339 莫队

    题意: 求任意一个区间的SG函数. 想到线段树,但是线段树合并很麻烦. 线段树——分块. 分块的一个应用就是莫队算法. 怎么暴力递推呢? 从一个区间到另一个区间,Ans 取决于 Ans 和 加入和删除 ...

  9. BZOJ 3236 莫队&plus;树状数组

    思路: 莫队+树状数组 (据说此题卡常数) yzy写了一天(偷笑) 复杂度有点儿爆炸 O(msqrt(n)logn) //By SiriusRen #include <cmath> #in ...

随机推荐

  1. 【流程管理】【PCB】PCB设计流程

    添加封装 封装库用官方库,如没有添加补丁库,用原库或其他库中元件复制修改 调用封装时可先放置到PCB里进行测量 3D模型添加网站 封装库分类按厂商分类,常用器件按器件类型分类, 命名使用规范 导入PC ...

  2. Jmockit使用

    引用单元测试中mock的使用及mock神器jmockit实践中的java单元测试中各种Mock框架对比,就能明白JMockit有多么强大: JMockit是基于JavaSE5中的java.lang.i ...

  3. suse linux 操作系统下打BASH补丁

    1.检查当前版本信息: bash -version echo $BASH_VERSION   2.打4.3版本的补丁 在tmp目录下(保险起见,空间至少要100M以上)新建一个bash_upgrade ...

  4. HDU4432 Sum of Divisors

    涉及知识点: 1. 进制转换. 2. 找因子时注意可以降低复杂度. Sum of divisors Time Limit: 2000/1000 MS (Java/Others)    Memory L ...

  5. 看大数据时代下的IT架构(1)业界消息队列对比

    一.MQ(Message Queue) 即 消息队列,一般用于应用系统解耦.消息异步分发,能够提高系统吞吐量.MQ的产品有很多,有开源的,也有闭源,比如ZeroMQ.RabbitMQ. ActiveM ...

  6. cocos2D(八)---- CCMenu &amp&semi;amp&semi;&amp&semi;amp&semi; CCMenuItem

    些菜单项让用户開始游戏.暂停\继续游戏.打开\关闭音乐或者是返回到上一个界面,比方以下两张图中用红色线框标记的菜单项     我们能够使用CCMenu和CCMenuItem实现上述的菜单功能,CCMe ...

  7. JMockit使用总结

    Jmockit可以做什么 使用JMockit API来mock被依赖的代码,从而进行隔离测试. 类级别整体mock和部分方法重写 实例级别整体mock和部分mock mock静态方法.私有变量.局部方 ...

  8. Bootstrap在线引用css和js

    百度在线调用 <script src="http://libs.baidu.com/bootstrap/3.0.3/js/bootstrap.min.js"></ ...

  9. 洛谷P1038 神经网络(bfs,模拟,拓扑)

    题目背景 人工神经网络(Artificial Neural NetworkArtificialNeuralNetwork)是一种新兴的具有自我学习能力的计算系统,在模式识别.函数逼近及贷款风险评估等诸 ...

  10. 来自工程师的8项Web性能提升建议

    在互联网盛行的今天,越来越多的在线用户希望得到安全可靠并且快速的访问体验.针对Web网页过于膨胀以及第三脚本蚕食流量等问题,Radware向网站运营人员提出以下改进建议,帮助他们为用户提供最快最优质的 ...