简要题意

给你 \(N\) 本书 \((h_i,w_i)\),你要将书分成任意段(顺序不能改变),使得每一段 \(j\) 中 \(\sum\limits_{i \in j} w_i \leq L\),段 \(j\) 的代价为 \(\max\limits_{i \in j}{h_i}\)。你需要输出每一段的代价之和的最小值。

\(1 \leq N \leq 10^{5}\)

思路

朴素 DP 思路

设 \(f_i\) 为前 \(i\) 本书的代价和。则:

\[\begin{aligned}
& W(i,j) = \sum_{k=i}^{j}{w_k} \\
& H(i,j) = \max_{k=i}^{j}{h_k} \\
& f_i = \min_{W(j,i) \leq L}{(f_{j-1} + H(j,i))}
\end{aligned}
\]

解释:将 \([j,i]\) 中的书放在合并,然后成为一个新的段。

时间复杂度 \(O(n^3)\)。经过前缀和优化后 \(O(n^2)\),无法通过本题。

代码如下:

for(int i=1;i<=n;i++){
dp[i]=0x7f7f7f7f7fll;
sum[i]=sum[i-1]+w[i];
}
for(int i=1;i<=n;i++){
mn=INT_MIN;
for(int j=i;j>0;j--){
mn=max(mn,h[j]);
if(sum[i]-sum[j-1]<=m){
dp[i]=min(dp[i],dp[j-1]+mn);
}
else{
break;
}
}
}
cout<<dp[n];

喜提 \(69\operatorname{pts}\)

DP 优化

首先,最后一个满足 \(W(j,i) \leq l\) 的 \(j\) 是可以二分得出的(因为题目中 \(w\) 的前缀和是单调不降的)。我们姑且设其为 \(p_i\),则:

\[f_i = \min_{p_i}^{i}{(f_{j-1} + H(j,i))}
\]

另外我们设 \(l_i\) 为左边第一个满足 \(h_j>h_i\) 的 \(j\),即:

\[l_i = \max_{j=1}^{n}{(j \cdot [h_j > h_i])}
\]

那么如果 \(l_i \lt j \leq i\),则 \(H(j,i)=h_i\),只需要找到最小的 \(f_{j-1}\)即可。至于其他的,就用继承下来的。这是正确的。

这道题没法 ODT,老老实实的线段树。

代码

耗时最长的一道题祭

#include <bits/stdc++.h>
#define ls (i<<1)
#define rs (i<<1|1)
#define mid ((l+r)>>1)
#define int long long
using namespace std; int f,n,l,h[100005],w[100005],sumw[100005]; namespace sgt{
struct node{
int f,fh,tag;
} t[400005];
void pushup(int i){
t[i].f=min(t[ls].f,t[rs].f);
t[i].fh=min(t[ls].fh,t[rs].fh);
}
void pushdown(int i){
if(t[i].tag!=(-1e9)){
t[ls].fh=t[ls].f+t[i].tag;
t[rs].fh=t[rs].f+t[i].tag;
t[ls].tag=t[i].tag;
t[rs].tag=t[i].tag;
t[i].tag=(-1e9);
}
}
void build(int i,int l,int r){
if(l==r){
t[i].fh=LLONG_MAX;
t[i].f=t[i].fh;
t[i].tag=(-1e9);
return;
}
build(ls,l,mid);
build(rs,mid+1,r);
pushup(i);
}
void init(int p,int i,int l,int r){
if(l==r){
t[i].fh=LLONG_MAX;
t[i].f=f;
return;
}
pushdown(i);
if(p<=mid){
init(p,ls,l,mid);
}
else{
init(p,rs,mid+1,r);
}
pushup(i);
}
void assign(int ql,int qr,int k,int i,int l,int r){
if(ql<=l&&r<=qr){
t[i].fh=t[i].f+k;
t[i].tag=k;
return;
}
pushdown(i);
if(ql<=mid){
assign(ql,qr,k,ls,l,mid);
}
if(mid<qr){
assign(ql,qr,k,rs,mid+1,r);
}
pushup(i);
}
int get(int ql,int qr,int i,int l,int r){
if(ql<=l&&r<=qr){
return t[i].fh;
}
pushdown(i);
int ret=LLONG_MAX;
if(mid>=ql){
ret=min(ret,get(ql,qr,ls,l,mid));
}
if(mid<qr){
ret=min(ret,get(ql,qr,rs,mid+1,r));
}
return ret;
}
} stack<int> sta;
int lft[100005]; signed main(){
cin>>n>>l;
for(int i=1;i<=n;i++){
cin>>h[i]>>w[i];
sumw[i]=sumw[i-1]+w[i];
}
h[0]=INT_MAX;sta.push(0);
for(int i=1;i<=n;i++){
while(!sta.empty()&&h[i]>h[sta.top()]){
sta.pop();
}
lft[i]=sta.top();
sta.push(i);
}
sgt::build(1,1,n);
for(int i=1;i<=n;i++){
// cout<<"INIT "<<i<<'\n';
sgt::init(i,1,1,n);
// cout<<"ASSIGN "<<i<<'\n';
sgt::assign(lft[i]+1,i,h[i],1,1,n);
// cout<<"LOWERBOUND "<<i<<'\n';
int ll = lower_bound(sumw,sumw+i+1,sumw[i]-l)-sumw;
if(ll<i){
f=sgt::get(ll+1,i,1,1,n);
}
}
cout<<f;
}

P1848 [USACO12OPEN]Bookshelf G的更多相关文章

  1. 2021.12.08 P1848 [USACO12OPEN]Bookshelf G(线段树优化DP)

    2021.12.08 P1848 [USACO12OPEN]Bookshelf G(线段树优化DP) https://www.luogu.com.cn/problem/P1848 题意: 当农夫约翰闲 ...

  2. p1848 [USACO12OPEN]书架Bookshelf

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

  3. 解题:USACO12OPEN Bookshelf

    题面 从零开始的DP学习之肆 当DP方程中的一部分具有某种单调性时可以用数据结构或者预处理维护来降低复杂度 一开始没有看懂题,尴尬,后来发现题目可以简化成这个样子: 将一个序列划分为若干段,每段长度不 ...

  4. pkuwc 前的任务计划

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

  5. Storyboards Tutorial 03

    这一节主要介绍segues,static table view cells 和 Add Player screen 以及 a game picker screen. Introducing Segue ...

  6. 文件图标SVG

    ​<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink ...

  7. [USACO12OPEN]书架Bookshelf

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

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

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

  9. [转]Linux下g++编译与使用静态库(.a)和动态库(.os) (+修正与解释)

    在windows环境下,我们通常在IDE如VS的工程中开发C++项目,对于生成和使用静态库(*.lib)与动态库(*.dll)可能都已经比较熟悉,但是,在linux环境下,则是另一套模式,对应的静态库 ...

随机推荐

  1. 齐博x1直播要设置回调地址才能播放

    因为通过扫码或推流网址给第三方用,也能让圈子实现直播,所以系统就改为必须要设置回调地址才能播放视频了.下面阿里与腾讯的都是大同小异的.腾迅的有多项,阿里的只有一项,不过阿里其实还有另一项,就是录制的时 ...

  2. Redis 02: redis基础知识 + 5种数据结构 + 基础操作命令

    Redis基础知识 1).测试redis服务的性能: redis-benchmark 2).查看redis服务是否正常运行: ping 如果正常---pong 3).查看redis服务器的统计信息: ...

  3. Linux三剑客sed

    注意sed和awk使用单引号,双引号有特殊解释 sed是Stream Editor(字符流编辑器)的缩写,简称流编辑器. sed是操作.过滤和转换文本内容的强大工具. 常用功能包括结合正则表达式对文件 ...

  4. python删除某一文件夹下的重复文件

    #2022-10-28 import hashlib import os import time def getmd5(filename): """ 获取文件 md5 码 ...

  5. Vue3 企业级优雅实战 - 组件库框架 - 2 初始化 workspace-root

    上文已经搭建了 pnpm + monorepo 的基础环境,本文对 workspace-root 进行初始化配置,包括:通用配置文件.公共依赖.ESLint. 1 通用配置文件 在项目 根目录 下添加 ...

  6. 【lwip】11-UDP协议&源码分析

    目录 前言 11.1 传输层说明 11.2 UDP协议简介 11.3 UDP特点 11.4 UDP端口号 11.5 UDP报文 11.6 UDP伪首部和校验和 11.7 wireshark报文分析 1 ...

  7. MessagePack 和System.Text.Json 序列号 反序列化对比

    本博客将测试MessagePack 和System.Text.Json 序列号 反序列化性能 项目文件: Program.cs代码: using BenchmarkDotNet.Running; us ...

  8. win10系统VMWare16 Pro 安装CentOS8

    目录 一.本机环境与问题解决 二.下载软件 三.VMWare16 Pro安装 四.CentOS8 安装 一.本机环境与问题解决 装了好几遍,感觉坑都踩了一遍,泪奔~,还好终于跑起来了! 查看电脑是否开 ...

  9. C ++:树

    C++:树 树的概念: 所谓"树"是输就结构的一种,树大概可以分为两大类: 有根树 和 无根树 有根树使有一个确定的根节点,反之为无根树 · 子节点:从树根开始,通过树边向下扩展的 ...

  10. c++题目:吃西瓜

    吃西瓜 [问题描述] 老胡买了是长方体形的西瓜来犒劳大家.... 这块西瓜长m厘米,宽n厘米,高h厘米.他发现如果把这块西瓜平均地分成m*n*h块1立方厘米的小正方体,那么每一小块都会有一个营养值(可 ...