题目大意
当Farmer John闲下来的时候,他喜欢坐下来读一本好书。 多年来,他已经收集了N本书 (1 <= N <= 100,000)。 他想要建立一个多层书架,来存放它们。 每本书 i 拥有一个宽度 W(i)和一个高度 H(i)。 所有的书需要按顺序,放到书架的每一层。 举例来说,第一层书架放k本书,应该放书1...k;第二层书架从第k+1本书开始放……。 每层书架的宽度最多为L (1 <= L <= 1,000,000,000)。 每层书架的高度为该层最高的那本书的高度。 书架的总高度为每层书架高度之和。 请帮FJ计算书架可能的最小总高度。
分析
一眼就看出是 dp (因为每天的第四或第三题都是dp)
最简单的方程是
f[i]=min{f[j]+max(h[j],h[j+1]...h[i])}
其中w[j]+w[j+1]...w[i]<=l。
但是这个东西是O(n²),会超时。所以联系到求最大值,可以想到用最大值的单调队列,队列里面存放的是以 i 为结尾,窗口里面的元素和恰好不大于 M 的 j 为左边界,即恰好有 sum[i] - sum[j-1] <= M.同时h也用单调队列优化。
然后这样一个可行解就是,f[ i ]=f[j]+h[a[r]](即队首元素的值), 这并不是最优解,所以还要找到队列中的最优解,一个可能的最优解只能是这样的f[i]=f[a[r-1]]+h[a[r]],也就是 a[ r ] 要大于后面的数,很显然,如果a[ r]小于后面的数,那么我们就可以将 a[ r ] 划分到后面去,而取得更优解。这里涉及的这个找最优解问题,可以用set。
看不懂的自行脑补(虽然我也不是很知道我在打什么,具体的见代码)
又是他—— 蜜汁c++
代码
#include <stdio.h>
#include <iostream>
#include <sstream>
#include <set>
using namespace std;
typedef long long ll;
ll que[100100],a[100100],dp[100100],b[100100];
ll n,m,sum;
ll i,j,k;
ll front,rear,p=1;
multiset <ll> lxf; //erase insert begin
int main()
{
scanf("%lld%lld",&n,&m);
for(i=1;i<=n;i++){
scanf("%lld%lld",&a[i],&b[i]);
}
front=1; rear=1;
dp[1]=a[1];que[rear]=1;
sum=b[1];
for(i=2;i<=n;i++){
sum+=b[i];
while(sum>m)
sum-=b[p++];
while( front<=rear&&a[i]>=a[que[rear]]){
lxf.erase(dp[que[rear-1]]+a[que[rear]]);
rear--;
}
rear=rear+1;
que[rear]=i;
if (front<rear) {
lxf.insert(dp[que[rear-1]]+a[que[rear]]);
}
while(que[front]<p&&front<=rear){
lxf.erase(dp[que[front]]+a[que[front+1]]);
front++;
}
dp[i]=dp[p-1]+a[que[front]];
j=*lxf.begin();
if ((j<dp[i])&&(j!=0)) dp[i]=j;
}
cout<<dp[n]<<endl;
return 0;
}