luogu 1969 积木大赛
题目链接
题意
初始序列为全\(0\),可以对序列进行的操作为将\([l,r]\)整体\(+1\),问操作多少次后可以得到序列\(a\).
思路
显然,最优的策略即是先找到整个序列的最小值,整体加上这么多,于是序列分成了两块;找到左半边的最小值,左半边整体加上;找到右半边的最小值,右半边整体加上……然而这样的做法是\(O(n^2)\)的,怎么办呢?
法一:线段树
用线段树维护区间最小值,区间修改+区间查询。
复杂度\(O(nlogn)\).
Code
#include <bits/stdc++.h>
#define lson (rt << 1)
#define rson (rt << 1 | 1)
#define inf 0x3f3f3f3f
#define maxn 100010
using namespace std;
typedef long long LL;
struct node {
int l, r, p, min, tag;
}tr[maxn * 4];
void push_up(int rt) {
if (tr[lson].min < tr[rson].min) tr[rt].min = tr[lson].min, tr[rt].p = tr[lson].p;
else tr[rt].min = tr[rson].min, tr[rt].p = tr[rson].p;
}
void build(int rt, int l, int r) {
tr[rt].l = l, tr[rt].r = r, tr[rt].min = inf;
if (l == r) {
scanf("%d", &tr[rt].min);
tr[rt].p = l;
return;
}
int mid = l + r >> 1;
build(lson, l, mid); build(rson, mid+1, r);
push_up(rt);
}
void push_down(int rt) {
if (tr[rt].tag) {
tr[lson].min += tr[rt].tag, tr[rson].min += tr[rt].tag;
tr[lson].tag += tr[rt].tag, tr[rson].tag += tr[rt].tag;
tr[rt].tag = 0;
}
}
void modify(int rt, int l, int r, int del) {
if (tr[rt].l == l && tr[rt].r == r) {
tr[rt].min += del;
tr[rt].tag += del;
return;
}
push_down(rt);
int mid = tr[rt].l + tr[rt].r >> 1;
if (r <= mid) modify(lson, l, r, del);
else if (l > mid) modify(rson, l, r, del);
else modify(lson, l, mid, del), modify(rson, mid+1, r, del);
push_up(rt);
}
int query(int rt, int l, int r, int& p) {
if (tr[rt].l == l && tr[rt].r == r) { p = tr[rt].p; return tr[rt].min; }
push_down(rt);
int mid = tr[rt].l + tr[rt].r >> 1;
if (r <= mid) return query(lson, l, r, p);
else if (l > mid) return query(rson, l, r, p);
else {
int p1, p2;
int ql = query(lson, l, mid, p1), qr = query(rson, mid+1, r, p2);
if (ql < qr) { p = p1; return ql; }
else { p = p2; return qr; }
}
}
int solve(int l, int r) {
if (l > r) return 0;
int p;
int mn = query(1, l, r, p);
if (l == r) return mn;
else {
if (p-1 >= l) modify(1, l, p-1, -mn);
if (p+1 <= r) modify(1, p+1, r, -mn);
return solve(l, p-1) + mn + solve(p+1, r);
}
}
int main() {
int n;
scanf("%d", &n);
build(1, 1, n);
printf("%d\n", solve(1, n));
return 0;
}
法二:换个视角
考虑第\(i\)块积木和第\(i+1\)块积木,如果第\(i+1\)块的积木低于或等于第\(i\)块积木的高度,那么在搭第\(i\)块积木的时候或者在这之前肯定就已经顺便搭好了第\(i+1\)块积木;而如果第\(i+1\)块的积木高于第\(i\)块积木的高度,那么第\(i+1\)块积木就必须要自己弥补上空缺的部分。
其实这个想法的具体操作完全等价于之前的思路,只不过是将每次找最小值的过程归到了从前往后的计算中,因此,在后一块积木的高度低于前一块积木的高度时无需再进行计算,因为已经在前面被计算过了。
Code
#include <bits/stdc++.h>
using namespace std;
typedef long long LL;
int main() {
int n;
scanf("%d", &n);
int p = 0, ans = 0;
for (int i = 0; i < n; ++i) {
int x;
scanf("%d", &x);
if (x > p) ans += x - p;
p = x;
}
printf("%d\n", ans);
return 0;
}
法三:分治
将左半边与右半边合并时,如果\(h[mid]\geq h[mid+1]\),则第\(mid+1\)块及之后被截断之前的若干块的\(h[mid+1]\)的部分都在搭第\(mid\)块时完成了;如果\(h[mid]\lt h[mid+1]\),则第\(mid+1\)块及之后被截断之前的若干块的\(h[mid]\)的部分也都在搭第\(mid\)块时完成了。
故solve(l,r)=solve(l,mid)+solve(mid+1,r)-min(h[mid],h[mid+1]);
Code
#include <bits/stdc++.h>
#define maxn 100010
using namespace std;
typedef long long LL;
int a[maxn];
int solve(int l, int r) {
if (l == r) return a[l];
int mid = l+r >> 1;
return solve(l, mid) + solve(mid+1, r) - min(a[mid], a[mid+1]);
}
int main() {
int n;
scanf("%d", &n);
for (int i = 1; i <= n; ++i) scanf("%d", &a[i]);
printf("%d\n", solve(1, n));
return 0;
}
luogu 1969 积木大赛的更多相关文章
- noip 2013 luogu P1969 积木大赛
题目描述 春春幼儿园举办了一年一度的“积木大赛”.今年比赛的内容是搭建一座宽度为n的大厦,大厦可以看成由n块宽度为1的积木组成,第i块积木的最终高度需要是hi. 在搭建开始之前,没有任何积木(可以看成 ...
- 洛谷—— P1969 积木大赛
https://www.luogu.org/problem/show?pid=1969 题目描述 春春幼儿园举办了一年一度的“积木大赛”.今年比赛的内容是搭建一座宽度为n的大厦,大厦可以看成由n块宽度 ...
- codevs 3288 积木大赛
题目描述 Description 春春幼儿园举办了一年一度的"积木大赛".今年比赛的内容是搭建一座宽度为 n 的大厦,大厦可以看成由 n 块宽度为1的积木组成,第i块积木的最终高度 ...
- NOIP2013积木大赛
题目描述 春春幼儿园举办了一年一度的“积木大赛”.今年比赛的内容是搭建一座宽度为n的大厦,大厦可以看成由n块宽度为1的积木组成,第i块积木的最终高度需要是hi. 在搭建开始之前,没有任何积木(可以看成 ...
- noip2013 积木大赛
题目描述 春春幼儿园举办了一年一度的“积木大赛”.今年比赛的内容是搭建一座宽度为n的大厦,大厦可以看成由n块宽度为1的积木组成,第i块积木的最终高度需要是hi. 在搭建开始之前,没有任何积木(可以看成 ...
- [NOIP2013] 提高组 洛谷P1969 积木大赛
题目描述 春春幼儿园举办了一年一度的“积木大赛”.今年比赛的内容是搭建一座宽度为n的大厦,大厦可以看成由n块宽度为1的积木组成,第i块积木的最终高度需要是hi. 在搭建开始之前,没有任何积木(可以看成 ...
- NOIp 2013 #1 积木大赛 Label:有趣的模拟
题目描述 春春幼儿园举办了一年一度的“积木大赛”.今年比赛的内容是搭建一座宽度为n的大厦,大厦可以看成由n块宽度为1的积木组成,第i块积木的最终高度需要是hi. 在搭建开始之前,没有任何积木(可以看成 ...
- AC日记——积木大赛 洛谷 P1969
题目描述 春春幼儿园举办了一年一度的“积木大赛”.今年比赛的内容是搭建一座宽度为n的大厦,大厦可以看成由n块宽度为1的积木组成,第i块积木的最终高度需要是hi. 在搭建开始之前,没有任何积木(可以看成 ...
- NOIP 2013 提高组 day2 积木大赛
积木大赛 描述 春春幼儿园举办了一年一度的“积木大赛”.今年比赛的内容是搭建一座宽度为 n 的大厦,大厦可以看成由 n 块宽度为1的积木组成,第
随机推荐
- Centos 6版本Device eth0 does not seem to be present,delaying initialization.故障处理
1.1 故障现象 2019年06月14日晚上,公司项目组说有台业务服务器连接不上,比较着急,我通过vpn拨入的方式远程登录到管理控制台查看发现网卡没有获取到IP地址,我尝试重启来重新启动,重启的时候 ...
- 43.VUE学习之--组件之使用.sync修饰符与computed计算属性超简单的实现美团购物车原理
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...
- 科学计算库Numpy——数值计算
矩阵 求和 乘积 最大值和最小值 最大值和最小值的位置 平均数 标准差 方差 限制 四舍五入
- A1095 Cars on Campus (30)(30 分)
A1095 Cars on Campus (30)(30 分) Zhejiang University has 6 campuses and a lot of gates. From each gat ...
- ACM二分搜索中的最大化最小值 总结
这类题目都有个相似的地方就是需要你去找一个临界点. 分析题目要你求什么,例如时间 那么mid就是时间 看求得这个跟什么相关 例如 poj 3258 求得是距离 这个距离跟两者之间的差相关 那题目要求你 ...
- spark测试几个hadoop的典型例子
1.求每年的最高温度数据格式如下: 0067011990999991950051507004888888889999999N9+00001+999999999999999999999900670119 ...
- bootstrap button
样式修改 .sign-button, .sign-button:hover, .sign-button:focus, .sign-button:active, .sign-button:visited ...
- day15 CSS JS DOM初探
居中 line-hight 是上下 text-line 是左右 实现一个返回顶部的功能: 1 先写好CSS 2 写动作JS 写一个悬浮菜单: <!DOCTYPE h ...
- IOS开发学习笔记019-动态创建控件
动态创建控件 一.按钮 二.文本输入框 三.lable标签 注意: 只是简单的拖拽控件会毁了你,所以最好还是手动通过代码创建控件. 如果要通过代码生成按钮的话,可以在系统自带的函数viewDidLoa ...
- python IDLE简介及使用技巧
前言:本人环境windows 7 64位,python2.7 IDLE简介: 是python 的可视化GUI编辑器 可以逐行输入命令 可方便的进行复制.粘贴等操作 常用命令行命令: import mo ...