P4644 [Usaco2005 Dec]Cleaning Shifts 清理牛棚
P4644 [Usaco2005 Dec]Cleaning Shifts 清理牛棚
你有一段区间需要被覆盖(长度 <= 86,399)
现有 \(n \leq 10000\) 段小线段, 每段可以从 \(l_{i}\) 到 \(r_{i}\) 花费为 \(s_{i}\)
求覆盖整个区间的最小花费
错误日志: 初始化时应该是 \(dp[L - 1]\) 为 \(0\) 而不是 \(dp[1]\) 为 \(0\) , 因为只需要覆盖 \([L, R]\)
Solution
设 \(dp[n]\) 表示从起点覆盖到 \(n\) 的最小花费
我们将小线段按照右端点升序排序, 满足无后效性
对于第 \(i\) 个线段, 有状态转移方程:$$dp[r_{i}] = min(dp[r_{i}], \min_{l_[i] - 1 \leq k < r_{i}}dp[k] + s_{i})$$
其含义为: 从第 \(i\) 段线段的起点开始选择上一个终点的最小值, 这样可以使线段相交(来保证无空白), 更新此线段能覆盖的终点
然后发现我们需要区间查询最小值和单点修改
为啥是单点修改不是区间修改呢? 当两个线段相交, 一定有 \(l_{x} \leq r_{y}\)
所以只需要在线段终点处单点修改即可
初始化 \(dp[L - 1]\) 为零
因为需要线段树维护, 尽量把起点设为 \(1\) 防止各种奇怪的错误, 本题所有位置点坐标 $ + 2$
还要注意处理一下超出 \([L, R]\) 的线段
Code
#include<iostream>
#include<cstdio>
#include<queue>
#include<cstring>
#include<algorithm>
#include<climits>
#define LL long long
#define REP(i, x, y) for(int i = (x);i <= (y);i++)
using namespace std;
int RD(){
int out = 0,flag = 1;char c = getchar();
while(c < '0' || c >'9'){if(c == '-')flag = -1;c = getchar();}
while(c >= '0' && c <= '9'){out = out * 10 + c - '0';c = getchar();}
return flag * out;
}
const int minn = 166419, INF = 1e9;
int num, L, R;
struct Node{
int l, r, s;
}I[minn];
bool cmp(Node a, Node b){return a.r < b.r;}
#define lid (id << 1)
#define rid (id << 1) | 1
int dp[minn];
struct seg_tree{
int l, r;
int min;
}tree[minn << 2];
void pushup(int id){tree[id].min = min(tree[lid].min, tree[rid].min);}
void build(int id, int l, int r){
tree[id].l = l, tree[id].r = r;
if(l == r){
tree[id].min = dp[l];
return ;
}
int mid = (l + r) >> 1;
build(lid, l, mid), build(rid, mid + 1, r);
pushup(id);
}
void update(int id, int val, int l, int r){
if(tree[id].l == l && tree[id].r == r){
tree[id].min = val;
return ;
}
int mid = (tree[id].l + tree[id].r) >> 1;
if(mid < l)update(rid, val, l, r);
else update(lid, val, l, r);
pushup(id);
}
int query(int id, int l, int r){
if(tree[id].l == l && tree[id].r == r)return tree[id].min;
int mid = (tree[id].l + tree[id].r) >> 1;
if(mid < l)return query(rid, l, r);
else if(mid >= r)return query(lid, l, r);
else return min(query(lid, l, mid), query(rid, mid + 1, r));
}
int main(){
num = RD(), L = RD() + 2, R = RD() + 2;//保证线段树左端点是1
REP(i, 1, R)dp[i] = INF;
dp[L - 1] = 0;
build(1, 1, R);
REP(i, 1, num){
I[i].l = RD() + 2;
I[i].r = RD() + 2;
I[i].s = RD();
I[i].l = I[i].l < L ? L : I[i].l;//处理一下超出范围的
I[i].r = I[i].r > R ? R : I[i].r;
}
sort(I + 1, I + 1 + num, cmp);
REP(i, 1, num){
int minn = query(1, I[i].l - 1, I[i].r);
//printf("minn=%d\n", minn);
dp[I[i].r] = min(dp[I[i].r], minn + I[i].s);
update(1, dp[I[i].r], I[i].r, I[i].r);
}
//REP(i, L - 1, R)printf("dp[%d]=%d\n", i - 2, dp[i]);
if(dp[R] == INF){puts("-1");return 0;}
printf("%d\n", dp[R]);
return 0;
}
P4644 [Usaco2005 Dec]Cleaning Shifts 清理牛棚的更多相关文章
- 洛谷P4644 [USACO2005 Dec]Cleaning Shifts 清理牛棚 [DP,数据结构优化]
题目传送门 清理牛棚 题目描述 Farmer John's cows, pampered since birth, have reached new heights of fastidiousness ...
- BZOJ1672: [Usaco2005 Dec]Cleaning Shifts 清理牛棚
1672: [Usaco2005 Dec]Cleaning Shifts 清理牛棚 Time Limit: 5 Sec Memory Limit: 64 MBSubmit: 414 Solved: ...
- BZOJ 1672: [Usaco2005 Dec]Cleaning Shifts 清理牛棚
题目 1672: [Usaco2005 Dec]Cleaning Shifts 清理牛棚 Time Limit: 5 Sec Memory Limit: 64 MB Description Farm ...
- BZOJ_1672_[Usaco2005 Dec]Cleaning Shifts 清理牛棚_动态规划+线段树
BZOJ_1672_[Usaco2005 Dec]Cleaning Shifts 清理牛棚_动态规划+线段树 题意: 约翰的奶牛们从小娇生惯养,她们无法容忍牛棚里的任何脏东西.约翰发现,如果要使这群 ...
- [Usaco2005 Dec]Cleaning Shifts 清理牛棚 (DP优化/线段树)
[Usaco2005 Dec] Cleaning Shifts 清理牛棚 题目描述 Farmer John's cows, pampered since birth, have reached new ...
- 【BZOJ1672】[Usaco2005 Dec]Cleaning Shifts 清理牛棚 动态规划
[BZOJ1672][Usaco2005 Dec]Cleaning Shifts Description Farmer John's cows, pampered since birth, have ...
- 【bzoj1672】[USACO2005 Dec]Cleaning Shifts 清理牛棚
题目描述 Farmer John's cows, pampered since birth, have reached new heights of fastidiousness. They now ...
- 【BZOJ】1672: [Usaco2005 Dec]Cleaning Shifts 清理牛棚(dp/线段树)
http://www.lydsy.com/JudgeOnline/problem.php?id=1672 dp很好想,但是是n^2的..但是可以水过..(5s啊..) 按左端点排序后 f[i]表示取第 ...
- 【bzoj1672】[USACO2005 Dec]Cleaning Shifts 清理牛棚 dp/线段树
题目描述 Farmer John's cows, pampered since birth, have reached new heights of fastidiousness. They now ...
随机推荐
- iOS开发学习-cocoapods的配置安装
安装coacoapods步骤: 在终端输入如下命令,升级ruby版本: sudo gem update —system 出现这个,标志着安装成功. 完成之后,再输入如下命令: gem sources ...
- Virtualbox+Ubuntu安装,VERR_VMX_MSR_ALL_VMX_DISABLED解决
学习链接:基于VirtualBox虚拟机安装Ubuntu图文教程--娄老师 启动虚拟机遇到的问题:BIOS中VT-x在所有CPU模式下被禁用(VERR_VMX_MSR_ALL_VMX_DISABLED ...
- BETA-7
前言 我们居然又冲刺了·七 团队代码管理github 站立会议 队名:PMS 530雨勤(组长) 过去两天完成了哪些任务 经过分析发现,为提升速率估测的可靠性,目前最具可改造性的参数为帧间间隔,调用参 ...
- Week 2
第1章:概论1.原文“这些软件企业的商业模式有些事合情合理也合法:有些看似合情合理,但不怎么合法:有些做法不合 理,但是还没有出台相关的法律.在相关法律完善之前,软件行业还有一个行规,即应该有职业道德 ...
- 词频统计的java实现方法——第一次改进
需求概要 原需求 1.读取文件,文件内包可含英文字符,及常见标点,空格级换行符. 2.统计英文单词在本文件的出现次数 3.将统计结果排序 4.显示排序结果 新需求: 1.小文件输入. 为表明程序能跑 ...
- 前端切图相关ps技术
标签(空格分隔): 前端切图 复制图层到一个新的ps文件 对于单个图层 1.选中图层 2.CTRL+A全选 3.CTRL+C 4.CTRL+N新建文件,文件大小默认就可以(背景透明也在这个面板设置), ...
- 微信小程序wx:for和wx:for-item的正确用法
wx:for="{{list}}"用来循环数组,而list即为数组名wx:for-item="items" 即用来定义一个循环过程中每个元素的变量的 如果是一维 ...
- C语言以字符形式读写文件
一.字符读取函数 fgetc (一).函数介绍 fgetc 是 file get char 的缩写,意思是从指定的文件中读取一个字符.函数原型为: int fgetc(FILE* fp) fp 为文件 ...
- Office/Visio/Project 2010 RTM (x86) (x64)(中文简体/英文)
1.cn_office_professional_plus_2010_x64_515528.exe ed2k://|file|cn_office_professional_plus_2010_x64_ ...
- c++11 委托构造
c++11 委托构造 #define _CRT_SECURE_NO_WARNINGS #include <iostream> #include <string> #includ ...