Description

当农夫约翰闲的没事干的时候,他喜欢坐下来看书。多年过去,他已经收集了 N 本书 (1 <= N <= 100,000), 他想造一个新的书架来装所有书。

每本书 i 都有宽度 W(i) 和高度 H(i)。书需要按顺序添加到一组书架上;比如说,第一层架子应该包含书籍1 ... k,第二层架子应该以第k + 1本书开始,以下如此。每层架子的总宽度最大为L(1≤L≤1,000,000,000)。每层的高度等于该层上最高的书的高度,并且整个书架的高度是所有层的高度的总和,因为它们都垂直堆叠。

请帮助农夫约翰计算整个书架的最小可能高度。

有N(1 <= N <= 100000)本书,每本书有一个宽度W(i),高度H(i),(1 <= H(i) <= 1,000,000; 1 <= W(i) <= L)。

现在有足够多的书架,书架宽度最多是L (1 <= L <= 1,000,000,000),把书按顺序(先放1,再放2.....)放入书架。某个书架的高度是该书架中所放的最高的书的高度。

将所有书放入书架后,求所有书架的高度和的最小值?

solution

弄了好久啊,首先这个DP没法维护啊,最后弄出一个暴力均摊的线段树做法.

\(f[i]=f[j]+v[j+1][i]\),这是原DP方程,\(v[j][i]\)表示\(j-i\)间的最大值.

考虑优化:

我们在线段树中,分别维护 \(f[j]\) 和 \(v[j]\),单调指针扫描,扫到 \(i\) 时,用 \(H[i]\) 取更新 \([1,i-1]\) 的 \(v\) 值.

再维护一个\(f[j]+v[j]\),那么转移就是线段树查最值了.

关键在于修改的复杂度:

维护一个区间最小值和区间最大值:

1.如果最小值大于 \(H[i]\),那么没有修改的必要,直接返回

2.如果最大值小于 \(H[i]\),那么直接打上覆盖标记即可

复杂度的证明:

一个无单调性的序列经过一次暴力修改之后,就会变成单调递增或递减序列了,那么下一次修改就会只会选择其中一部分进行修改(因为另一部分总会碰到上述两个剪枝中的一种情况),并且修改完之后依旧是满足单调性的,所以除了第一次修改之外,之后就是线段树修改的复杂度了,均摊 \(O(n*logn)\),常数有些大

#include <algorithm>
#include <iostream>
#include <cstdlib>
#include <cstring>
#include <cstdio>
#include <cmath>
#define RG register
#define ls (o<<1)
#define rs (o<<1|1)
#define Min(a,b) ((a)<(b)?(a):(b))
#define Max(a,b) ((a)>(b)?(a):(b))
using namespace std;
typedef long long ll;
const int N=100005;const ll inf=1e15;
int n,m,v[N],lazy[N<<2],mx[N<<2],mn[N<<2];ll f[N<<2],a[N],g[N<<2];
inline void upd(RG int o){
f[o]=Min(f[ls],f[rs]);
mx[o]=Max(mx[ls],mx[rs]);
mn[o]=Min(mn[ls],mn[rs]);
}
inline void pushdown(RG int o){
if(!lazy[o])return ;
int k=lazy[o];
f[ls]=g[ls]+k;mx[ls]=k;mn[ls]=k;
f[rs]=g[rs]+k;mx[rs]=k;mn[rs]=k;
lazy[ls]=k;lazy[rs]=k;
lazy[o]=0;
}
inline void Modify(int l,int r,int o,int sa,int se,int t){
if(l!=r)pushdown(o);
if(mn[o]>=t)return ;
if(sa<=l && r<=se){
if(mx[o]<t){
lazy[o]=t;f[o]=g[o]+t;mx[o]=mn[o]=t;
return ;
}
}
if(l==r)return ;
int mid=(l+r)>>1;
if(se<=mid)Modify(l,mid,ls,sa,se,t);
else if(sa>mid)Modify(mid+1,r,rs,sa,se,t);
else Modify(l,mid,ls,sa,mid,t),Modify(mid+1,r,rs,mid+1,se,t);
upd(o);
}
inline void updata(int l,int r,int o,int sa,ll t){
if(l==r){g[o]=t;return ;}
int mid=(l+r)>>1;
pushdown(o);
if(sa<=mid)updata(l,mid,ls,sa,t);
else updata(mid+1,r,rs,sa,t);
g[o]=Min(g[ls],g[rs]);
}
inline ll qry(int l,int r,int o,int sa,int se){
if(l!=r)pushdown(o);
if(sa<=l && r<=se)return f[o];
int mid=(l+r)>>1;ll ret,q1,q2;
if(se<=mid)ret=qry(l,mid,ls,sa,se);
else if(sa>mid)ret=qry(mid+1,r,rs,sa,se);
else{
q1=qry(l,mid,ls,sa,mid),q2=qry(mid+1,r,rs,mid+1,se);
ret=Min(q1,q2);
}
upd(o);
return ret;
}
int l=0;
void work()
{
scanf("%d%d",&n,&m);
for(int i=1;i<=n;i++)
scanf("%d%lld",&v[i],&a[i]),a[i]+=a[i-1];
ll tmp;
for(int i=1;i<=n;i++){
while(l<i && a[i]-a[l]>m)l++;
Modify(0,n,1,0,i-1,v[i]);
tmp=qry(0,n,1,l,i-1);
if(i!=n)updata(0,n,1,i,tmp);
}
cout<<tmp<<endl;
} int main()
{
freopen("pp.in","r",stdin);
freopen("pp.out","w",stdout);
work();
return 0;
}

[USACO12OPEN]书架Bookshelf的更多相关文章

  1. [Luogu1848][USACO12OPEN]书架Bookshelf DP+set+决策单调性

    题目链接:https://www.luogu.org/problem/show?pid=1848 题目要求书必须按顺序放,其实就是要求是连续的一段.于是就有DP方程$$f[i]=min\{f[j]+m ...

  2. p1848 [USACO12OPEN]书架Bookshelf

    分析 单调队列优化dp即可 正确性显然,详见代码 代码 #include<bits/stdc++.h> using namespace std; #define int long long ...

  3. 书架 bookshelf

    书架 bookshelf 题目描述 当Farmer John闲下来的时候,他喜欢坐下来读一本好书. 多年来,他已经收集了N本书 (1 <= N <= 100,000). 他想要建立一个多层 ...

  4. pkuwc 前的任务计划

    菜鸡 wxw 的计划(肯定会咕咕咕 12.27 luogu P4244 [SHOI2008]仙人掌图 II(咕咕咕 luogu P4246 [SHOI2008]堵塞的交通 (没有咕! luogu P1 ...

  5. Java设计模式学习记录-迭代器模式

    前言 这次要介绍的是迭代器模式,也是一种行为模式.我现在觉得写博客有点应付了,前阵子一天一篇,感觉这样其实有点没理解透彻就写下来了,而且写完后自己也没有多看几遍,上次在面试的时候被问到java中的I/ ...

  6. English trip -- VC(情景课)10 B Around the house 在家里

    Vocablulary focus  核心词汇 cook play the guitar listen to music watch TV read magazines work in the gar ...

  7. Python之Django的Model详解

    一.创建数据库 创建数据库 进入数据库: mysql -uroot -p 创建数据库: CREATE DATABASE test1 CHARSET=utf8; 连接数据库 虚拟环境中安装数据库模块:p ...

  8. 设计模式——迭代器(Iterator)模式

    概述 迭代器模式简单的说(按我目前的理解)就是一个类提供一个对外迭代的接口,方面调用者迭代.这个迭代接口至少包括两个方法:hasNext()--用于判断是否还有下一个,next()--用于取出下一个对 ...

  9. XStream的基本使用

    先准备两个bean public class Book { private int bookId; private String bookName; private String bookCode; ...

随机推荐

  1. 每日冲刺报告-Day4

    敏捷冲刺报告--Day4 情况简介 今天完成前端后端任务对接, GUI主体编写 任务进度 赵坤: 完成后端爬虫 李世钰: 前后端对接, GUI编写 黄亦薇:召集小组成员开会,帮助查找资料,寻找BUG ...

  2. OO前三次作业总结

    一.第一次作业 1.程序设计分析 ![img](s1.ax1x.com/2018/04/02/CSgoSU.png) 图1 第一次作业类图 ![name](https://images2018.cnb ...

  3. nyoj 星期几?

    星期几? 时间限制:500 ms  |  内存限制:65535 KB 难度:2   描述                      Acmer 小鱼儿 埋头ku算一道题 条件:已知给定 一日期 告诉你 ...

  4. XML之自动生成类,添加,修改,删除类的属性

    1. class ClassHelperDemo { public static void Main() { #region 演示一:动态生成类. //生成一个类t. Type t = ClassHe ...

  5. JAVA_SE基础——54.异常

    异常的体系: ----------| Throwable  所以异常或者错误类的超类 --------------|Error  错误   错误一般是用于jvm或者是硬件引发的问题,所以我们一般不会通 ...

  6. ajax 操作

    ajax 操作 ajax呢,就是要做到在神不知鬼不觉的情况之下给服务端发送请求. ajax能干啥哩? 这,,,,: 利用AJAX可以做:1.注册时,输入用户名自动检测用户是否已经存在.2.登陆时,提示 ...

  7. 快速搭建ssm框架

    快速搭建SSM框架 因为最近有很多朋友问我自己的项目搭建的不够完善,并且经常出现一些小问题,那么今天我又整理了一下文档教大家如何快速搭建SSM框架我是用 eclipse搭建的,如果想用idear的话我 ...

  8. Spring入门(3-1)Spring的标签命名空间

    1.标签命名空间声明: 2.标签命名空间使用 标签默认的命名空间是 security:,可以不用带 security:,直接写标签,如: <http  <authentication-ma ...

  9. 通过wget工具下载指定文件中的URLs对应的资源并保存到指定的本地目录中去并进行文件完整性与可靠性校验

    创建URLs文件在终端输入cd target_directory回车,便把当前文件夹切换到了目标文件夹target_directory,此后创建的文件都会丢它里面在终端输入cat > URLs回 ...

  10. ubuntu16.04下安装chrome

    1.在终端中,输入以下命令: sudo wget http://www.linuxidc.com/files/repo/google-chrome.list -P /etc/apt/sources.l ...