书架 bookshelf
书架 bookshelf
题目描述
当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计算书架可能的最小总高度。
输入
第1行:两个空格隔开的整数:N和L
第2行..第N+1行:第i+1行包含两个空格隔开的整数:H(i) 和 W(i) 。(1 <= H(i) <= 1,000,000; 1 <= W(i) <= L)。
输出
仅一行,表示书架可能的最小总高度
样例输入
5 10
5 7
9 2
8 5
13 2
3 8
样例输出
21
提示
【样例解释】
一共有5本书。每层书架的宽度最多为10。
3层书架,第1层放书1(高5,宽7),第2层放书2..4(高13,宽9),第3层放书5(高3,宽8)。
【数据范围】
45%的数据:1<=N<=5,000
100%的数据:1<=N<=100,000,1<=Wi<=L<=1,000,000,000,1<=Hi<=1,000,000
solution
考虑暴力dp
f[i]表示放好前i本书的最小高度。
f[i]=min (f[j]+max(h[j+1]--h[i])) (w[i]-w[j]<=L)
效率O(n^2)
现在优化时间复杂度。
枚举右端点r,我们维护最靠前的可转移位置l。
也就是我要知道[l,r]内f[i]+maxh[i~r]的最小值
由于f不变,h可能变(而且是整段改变),我们用线段树维护。
f相当于单点加,而h则是区间修改。
分别维护f的最小值和f+h的最小值,改变h时用旧的f与新的h更新f+h即可。
#include<cstdio>
#include<iostream>
#include<cstdlib>
#include<cstring>
#include<algorithm>
#include<cmath>
#define maxn 100005
#define inf 1e16
#define ll long long
using namespace std;
int n,id[maxn],top;
ll w[maxn],h[maxn],zh[maxn],len,fn;
struct node{
ll f,bj,mi;
}tr[maxn*];
void update(int k,ll v){
tr[k].bj=v;tr[k].mi=tr[k].f+v;
}
void down(int k){
if(tr[k].bj){
ll &v=tr[k].bj;
update(k*,v);update(k*+,v);
v=;
}
}
void wh(int k){
tr[k].f=min(tr[k*].f,tr[k*+].f);
tr[k].mi=min(tr[k*].mi,tr[k*+].mi);
}
void build(int k,int l,int r){
if(l==r){
if(l) tr[k].f=tr[k].mi=inf;
return;
}
int mid=l+r>>;
build(k*,l,mid);build(k*+,mid+,r);
wh(k);
}
void ch(int k,int l,int r,int li,int ri,ll v){
if(l>=li&&r<=ri){
update(k,v);return;
}
down(k);
int mid=l+r>>;
if(li<=mid)ch(k*,l,mid,li,ri,v);
if(ri>mid)ch(k*+,mid+,r,li,ri,v);
wh(k);
}
ll ask(int k,int l,int r,int li,int ri){
if(l>=li&&r<=ri){
return tr[k].mi;
}
down(k);
int mid=l+r>>;
ll tmp=inf;
if(li<=mid)tmp=min(tmp,ask(k*,l,mid,li,ri));
if(ri>mid)tmp=min(tmp,ask(k*+,mid+,r,li,ri));
return tmp;
}
void add(int k,int l,int r,int pl,ll v){
if(l==r){ tr[k].f=v;return;
}
down(k);
int mid=l+r>>;
if(pl<=mid)add(k*,l,mid,pl,v);
else add(k*+,mid+,r,pl,v);
wh(k);
}
int main(){ cin>>n>>len;
for(int i=;i<=n;i++){
scanf("%lld%lld",&h[i],&w[i]);
w[i]+=w[i-];
}
int la=;
build(,,n);
for(int i=;i<=n;i++){
while(top&&h[i]>zh[top])top--;
//cout<<"ch "<<id[top]<<' '<<i-1<<endl;
ch(,,n,id[top],i-,h[i]);
while(w[i]-w[la]>len)la++;
//cout<<"la "<<la<<endl;
fn=ask(,,n,la,i-);
//cout<<fn<<endl;
add(,,n,i,fn);
zh[++top]=h[i];id[top]=i;
}
cout<<fn<<endl;
return ;
}
书架 bookshelf的更多相关文章
- [USACO12OPEN]书架Bookshelf
Description 当农夫约翰闲的没事干的时候,他喜欢坐下来看书.多年过去,他已经收集了 N 本书 (1 <= N <= 100,000), 他想造一个新的书架来装所有书. 每本书 i ...
- [Luogu1848][USACO12OPEN]书架Bookshelf DP+set+决策单调性
题目链接:https://www.luogu.org/problem/show?pid=1848 题目要求书必须按顺序放,其实就是要求是连续的一段.于是就有DP方程$$f[i]=min\{f[j]+m ...
- p1848 [USACO12OPEN]书架Bookshelf
分析 单调队列优化dp即可 正确性显然,详见代码 代码 #include<bits/stdc++.h> using namespace std; #define int long long ...
- Java设计模式学习记录-迭代器模式
前言 这次要介绍的是迭代器模式,也是一种行为模式.我现在觉得写博客有点应付了,前阵子一天一篇,感觉这样其实有点没理解透彻就写下来了,而且写完后自己也没有多看几遍,上次在面试的时候被问到java中的I/ ...
- 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 ...
- Python之Django的Model详解
一.创建数据库 创建数据库 进入数据库: mysql -uroot -p 创建数据库: CREATE DATABASE test1 CHARSET=utf8; 连接数据库 虚拟环境中安装数据库模块:p ...
- pkuwc 前的任务计划
菜鸡 wxw 的计划(肯定会咕咕咕 12.27 luogu P4244 [SHOI2008]仙人掌图 II(咕咕咕 luogu P4246 [SHOI2008]堵塞的交通 (没有咕! luogu P1 ...
- 设计模式——迭代器(Iterator)模式
概述 迭代器模式简单的说(按我目前的理解)就是一个类提供一个对外迭代的接口,方面调用者迭代.这个迭代接口至少包括两个方法:hasNext()--用于判断是否还有下一个,next()--用于取出下一个对 ...
- XStream的基本使用
先准备两个bean public class Book { private int bookId; private String bookName; private String bookCode; ...
随机推荐
- [文章存档]Azure .net WebAPP的js/css文件过大导致访问慢的解决办法
https://docs.azure.cn/zh-cn/articles/azure-operations-guide/app-service-web/aog-app-service-web-qa-j ...
- 第二次作业(homework-02)成绩公布
学位后三位和对应成绩: 057 0008 4011 4012 7014 5015 5017 6018 0019 0026 2027 7036 0038 7.5046 7048 6.5051 0061 ...
- Fifteen scrum meeting 2015-11-21
最近几日因为其他作业着实拖延了很久更新工程进度. 闫昊: 完成:学习讨论区开发 即将进行:讨论区代码开发 唐彬: 完成:学习学习进度部分开发 即将进行:学习进度功能开发 史烨轩: 完成:学习下载功能设 ...
- 使用sqlyog创建数据库的错误
1.错误代码: 1064 You have an error in your SQL syntax; check the manual that corresponds to your MySQL s ...
- Task 6.2冲刺会议十 /2015-5-23
今天是第一个冲刺阶段的最后一天,主要把做出来的程序进行了初步的测试,在一台笔记本上运行程序,摄像头可以工作也能听到声音和麦克多的运转也还可以,两台计算机同时在一个局域网中通信的时候也可以实现.不过后续 ...
- (2016.2.2)1001.A+B Format (20)解题思路
https://github.com/UNWILL2LOSE/object-oriented 解题思路 目标: *首先运算要求实现输入2个数后,输出类似于银行的支票上的带分隔符规则的数字. 代码实现思 ...
- 作业6 团队项目之需求 (NABCD模型)
N A B C D模型分析 WorkGroup:NewApps 组员:欧其锋(201306114305 http://www.cnblogs.com/ouqifeng/) 吕日荣(20130611 ...
- WebGL学习笔记(二)
目录 绘制多个顶点 使用缓冲区对象 类型化数组 使用drawArrays()函数绘制图形 图形的移动 图形的旋转 图形的缩放 绘制多个顶点 使用缓冲区对象 创建缓冲区对象 var vertexBuff ...
- js对象进行浅复制,深拷贝的方法
js对象浅拷贝和深拷贝详解 本文为大家分享了JavaScript对象的浅拷贝和深拷贝代码,供大家参考,具体内容如下 1.浅拷贝 拷贝就是把父对像的属性,全部拷贝给子对象. 下面这个函数,就是在做拷 ...
- sql学习. case + group by 都干了啥子事情
select case pref_name when 'fudao' then 'siguo' when 'xiangchuan' then 'siguo' when 'aiyuan' then 's ...