书架_纪中2931_dp_堆_单调队列

题目大意

  当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;  
}  

posted @ 2016-07-10 10:49  一个响亮的蒟蒻  阅读(192)  评论(0编辑  收藏  举报