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的积木组成,第
随机推荐
- thinkcmf5 iis+php重写配置
TP在本机运行非常好,谁想到服务器上后,连http://www.***.com/wap/login/index都404错误了, 中间的郁闷过程不表. 解决方案分两步: 第一步: 下载rewrite_2 ...
- Jin Ge Jin Qu hao UVA - 12563 01背包
题目:题目链接 思路:由于t最大值其实只有180 * 50 + 678,可以直接当成01背包来做,需要考虑的量有两个,时间和歌曲数,其中歌曲优先级大于时间,于是我们将歌曲数作为背包收益,用时间作为背包 ...
- [Poj3133]Manhattan Wiring (插头DP)
Description 题目大意:给你个N x M(1≤N, M≤9)的矩阵,0表示空地,1表示墙壁,2和3表示两对关键点.现在要求在两对关键点之间建立两条路径,其中两条路径不可相交或者自交(就是重复 ...
- [Hdu1693]Eat the Trees(插头DP)
Description 题意:在n*m(1<=N, M<=11 )的矩阵中,有些格子有树,没有树的格子不能到达,找一条或多条回路,吃完所有的树,求有多少种方法. Solution 插头DP ...
- readhat7.0 bond配置
Bonding的模式一共有7种: 1.mode=0(balance-rr)(平衡抡循环策略) 概念:链路负载均衡,增加带宽,支持容错,一条链路故障会自动切换正常链路.交换机需要配置聚合口,思科叫por ...
- HDU 5739 Fantasia 双连通分量 树形DP
题意: 给出一个无向图,每个顶点有一个权值\(w\),一个连通分量的权值为各个顶点的权值的乘积,一个图的权值为所有连通分量权值之和. 设删除顶点\(i\)后的图\(G_i\)的权值为\(z_i\),求 ...
- Three Steps to Migrate Group Policy Between Active Directory Domains or Forests Using PowerShell
Three Steps Ahead Have you ever wished that you had three legs? Imagine how much faster you could ru ...
- CentOS6.5创建yum源
昨天给布置个新的需求,做一个Yum仓库,要求是HTTP式的,在某个服务器上搭建个Yum仓库,能让其它的机器有了这个机器的.repo仓库文件后就可以从本地下载安装软件,以前都是下载后直接yum inst ...
- linux环境搭建系列之memcached安装步骤
1.从官网在线下载最新的安装包 wget http://memcached.org/downloads/memcached-1.4.34.tar.gz 该命令为在线下载 注意:最新的地址会变动,所以最 ...
- sqlserver执行时间和自行效率
SET STATISTICS PROFILE ON --SET STATISTICS IO ON --SET STATISTICS TIME ON declare @dtm datetime SQL语 ...