【树状数组 思维题】luoguP3616 富金森林公园
树状数组、差分、前缀和、离散化
题目描述
博艾的富金森林公园里有一个长长的富金山脉,山脉是由一块块巨石并列构成的,编号从1到N。每一个巨石有一个海拔高度。而这个山脉又在一个盆地中,盆地里可能会积水,积水也有一个海拔高度,所有严格低于这个海拔高度的巨石,就会在水面下隐藏。
由于地壳运动,巨石的海拔高度可能会随时变化,每次一块的巨石会变成新的海拔高度。当然,水面的高度也会随时发生变化。
因为有这样奇妙的地质奇观,吸引了很多游客来游玩。uim作为一个游客,可以告诉你此时水位海拔,你得告诉他,能看到有几个连续露出水面的部分。(与水面持平我们也认为是露出)
输入输出格式
输入格式:
第一行两个整数N和M,分别表示N块石头,M个询问。
接下来一行,N个整数Ai表示每个巨石的初始海拔。
接下来M行,每行有两个或者三个数,每一行如果第一个数是1,那么后面跟一个Bj,表示水面海拔。如果第一个数是2,后面跟两个整数,Cj和Dj,表示编号Cj的巨石海拔变为Dj。
输出格式:
对于每个"1"询问,给出一个整数答案,也就是露出了几部分的山峰。
说明
10%的数据, N,M<=2000
另外30%的数据, 只有"1"的询问。
100%的数据, 1<=N,M<=200000,1<=Ai,Bj,Dj<=10^9,一定有"1"询问
题目分析
首先考虑暴力。每一次直接修改,查询时候$O(n)$查询,总复杂度$O(NM)$这样能拿50分。比赛做的时候有想过用树状数组维护,但是由于每次高度查询都不相同,因此没有搞出来。
后来去请教YKH,也看了看他的代码。这题用树状数组维护前缀和没有错,不过还要涉及到差分和离散化的应用。
容易发现在高度已知的情况下,对于相邻两个元素$f[i] > f[i-1]$,当且仅当$f[i-1] < query-height < f[i]$时对答案有贡献。那么可以看出,对于一组$f[i-1],f[i]$,它对任意$f[i-1] < height < f[i]$都有价值为1的贡献。显而易见的,这个特性使得我们可以用线性数据结构维护它。而差分+单点查询区间修改的树状数组是不错的选择(当然线段树也可以并且思维复杂度略小,不过比有差分的树状数组要慢不少)。
我们先将操作行为全部读入,考虑离线做法。
看到数据范围就知道直接用下标作为高度是会原地爆炸的。所以先将原始点和操作点都先存储起来,再对它们一起离散化。这样就可以避免已经离散化的一个数列例如{$1,3,5,7(原始)$}→{$1,2,3,4(离散化)$}再修改出未曾安排过离散的数例如{$1,3,5,7(原始)$}→{$1,3,5,4(修改后)$}→{$1,2,3,?(此时离散化爆炸)$}。
所有数据都读入之后,再对所有点按高度排序(不管它本身的属性是询问点还是修改点)。接下去先对点集手工离散化(不用unique的原因是这里不同类型的点离散化操作不同)。
离散化后我们拥有的数据应该是这样的:
$Q[i][0..2] i≤m$ 记录所有操作
$按照高度排序的s[]$ 记录所有点
数据既然已经预处理好了,接下去就是先对$tot$个点处理一遍了。处理也就不过是按照差分的思路单点修改。
处理操作时,对于query我们直接用树状数组对于查询点离散后的高度求前缀和;对于update我们要先考虑修改前的点是否与相邻的点产生贡献。因为这里的历史性是连续的(当前状况只从上一次操作转移过来),所以差分可以轻松完成撤销操作——只要根据原数据的逆操作即可。同理,更新过后再在左右检查一遍是否对答案有贡献,这样复杂度就是有保证的了。实测此方法540ms可过。
上YKH的代码(注释是我学习他代码过程中边看边打的)
#include <cstdio>
#include <algorithm>
using namespace std; int read()
{
char c;while(c=getchar(),c<''||c>'');
int x=c-'';while(c=getchar(),c>=''&&c<='')x=x*+c-'';
return x;
} int N,M,tot,x,y,Q[][],H[]; struct wx{
int x,y,c;//c-0原先高度 c-1询问 c-2修改操作
}S[]; int cmp(wx x,wx y){return x.x<y.x;} int ar[]; void add(int x,int y)
{
for(int i=x;i<=tot;i+=i&-i){
ar[i]+=y;
}
return ;
} int get(int x)
{
int tot=;
for(int i=x;i>;i-=i&-i){
tot+=ar[i];
}
return tot;
} int main()
{
// freopen("x.txt","r",stdin);
// freopen("w.txt","w",stdout);
N=read(),M=read();
register int i,j;
for(i=;i<=N;i++)x=read(),S[++tot]=(wx){x,i,};
for(i=;i<=M;i++){
Q[i][]=read(),y=read();
if(Q[i][]==){
Q[i][]=y;y=read();
S[++tot]=(wx){y,i,};
continue;
}
S[++tot]=(wx){y,i,};
} //Q[i][0]表示操作的类型 sort(S+,S+tot+,cmp);
S[].x=-2e9;
int col=;
for(i=;i<=tot;i++){
if(S[i].x!=S[i-].x)col++;
if(S[i].c==)Q[S[i].y][]=col;
else if(S[i].c==)Q[S[i].y][]=col;
else H[S[i].y]=col;//对H[]离散化
}
for(i=;i<=N;i++){
if(H[i]>H[i-]){
add(H[i-]+,);
add(H[i]+,-);
}
}
for(i=;i<=M;i++){
if(Q[i][]==)printf("%d\n",get(Q[i][]));
else {//离线update
if(H[Q[i][]]>H[Q[i][]-]){
add(H[Q[i][]-]+,-);
add(H[Q[i][]]+,);
}
if(Q[i][]<N && H[Q[i][]]<H[Q[i][]+]){
add(H[Q[i][]]+,-);
add(H[Q[i][]+]+,);
}
H[Q[i][]]=Q[i][];
if(H[Q[i][]]>H[Q[i][]-]){
add(H[Q[i][]-]+,);
add(H[Q[i][]]+,-);
}
if(Q[i][]<N && H[Q[i][]]<H[Q[i][]+]){
add(H[Q[i][]]+,);
add(H[Q[i][]+]+,-);
}
}
}
return ;
}
这是我的代码(自认为可读性挺高emm)
#include<bits/stdc++.h>
using namespace std;
const int N = *1e5+;
const int M = *1e5+;
struct point
{
int x,y,c;
}s[N+M];
int Q[M][],height[*N],dl[*N],n,m,tot;
inline int lowbit(int x){return x&-x;}
inline void add(int x, int c){for (; x<=tot; x+=lowbit(x))dl[x]+=c;}
inline int query(int x)
{
int ret = ;
for (; x; x-=lowbit(x))ret+=dl[x];
return ret;
}
inline int read()
{
int x = ;char c = getchar();
while (c<''||c>'')c = getchar();
while (c>=''&&c<=''){x = x*+c-;c = getchar();}
return x;
}
bool cmp(point a, point b){return a.x<b.x;}
int main()
{
n = read();m = read();
for (int i=; i<=n; i++)s[++tot] = (point){read(), i, };
for (int i=; i<=m; i++)
{
Q[i][] = read();
int y = read();
if (Q[i][]-){
Q[i][] = y;y = read();
s[++tot] = (point){y, i, };
continue;
}
s[++tot] = (point){y, i, };
}
sort(s+, s+tot+, cmp);
int col = ;
s[].x = -*1e9;
for (int i=; i<=tot; i++)
{
if (s[i].x!=s[i-].x)col++;
if (s[i].c==)Q[s[i].y][] = col;
else if (s[i].c==)Q[s[i].y][] = col;
else height[s[i].y] = col;
}
for (int i=; i<=n; i++)
{
if (height[i]>height[i-])
{
add(height[i]+, -);
add(height[i-]+, );
}
}
for (int i=; i<=m; i++)
{
if (-Q[i][]){printf("%d\n",query(Q[i][]));continue;}
int qs = Q[i][];
if (height[qs-]<height[qs])
{
add(height[qs]+, );
add(height[qs-]+, -);
}
if (qs < n&&height[qs]<height[qs+])
{
add(height[qs+]+, );
add(height[qs]+, -);
}
height[qs] = Q[i][];
if (height[qs-]<height[qs])
{
add(height[qs]+, -);
add(height[qs-]+, );
}
if (qs < n&&height[qs]<height[qs+])
{
add(height[qs+]+, -);
add(height[qs]+, );
}
}
return ;
}
【树状数组 思维题】luoguP3616 富金森林公园的更多相关文章
- 树状数组 洛谷P3616 富金森林公园
P3616 富金森林公园 题目描述 博艾的富金森林公园里有一个长长的富金山脉,山脉是由一块块巨石并列构成的,编号从1到N.每一个巨石有一个海拔高度.而这个山脉又在一个盆地中,盆地里可能会积水,积水也有 ...
- HDU 1166 敌兵布阵(线段树/树状数组模板题)
敌兵布阵 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others) Total Submi ...
- st表树状数组入门题单
预备知识 st表(Sparse Table) 主要用来解决区间最值问题(RMQ)以及维护区间的各种性质(比如维护一段区间的最大公约数). 树状数组 单点更新 数组前缀和的查询 拓展:原数组是差分数组时 ...
- bzoj1103树状数组水题
(卧槽,居然规定了修改的两点直接相连,亏我想半天) 非常水的题,用dfs序(而且不用重复,应该是直接规模为n的dfs序)+树状数组可以轻松水 收获:树状数组一遍A(没啥好骄傲的,那么简单的东西) #i ...
- UESTC 1584 Washi与Sonochi的约定【树状数组裸题+排序】
题目链接:UESTC 1584 Washi与Sonochi的约定 题意:在二维平面上,某个点的ranked被定义为x坐标不大于其x坐标,且y坐标不大于其y坐标的怪物的数量.(不含其自身),要求输出n行 ...
- 敌兵布阵 HDU - 1166 (树状数组模板题,线段树模板题)
思路:就是树状数组的模板题,利用的就是单点更新和区间求和是树状数组的强项时间复杂度为m*log(n) 没想到自己以前把这道题当线段树的单点更新刷了. 树状数组: #include<iostrea ...
- 树状数组训练题1:弱弱的战壕(vijos1066)
题目链接:弱弱的战壕 这道题似乎是vijos上能找到的最简单的树状数组题了. 原来,我有一个错误的思想,我的设计是维护两个树状数组,一个是横坐标,一个是纵坐标,然后读入每个点的坐标,扔进对应的树状数组 ...
- 树状数组 简单题 cf 961E
题目链接 : https://codeforces.com/problemset/problem/961/E One day Polycarp decided to rewatch his absol ...
- Lightoj 1112 - Curious Robin Hood 【单点改动 + 单点、 区间查询】【树状数组 水题】
1112 - Curious Robin Hood PDF (English) Statistics Forum Time Limit: 1 second(s) Memory Limit: 64 MB ...
随机推荐
- C 语言实例 - 查找字符在字符串中出现的次数
C 语言实例 - 查找字符在字符串中出现的次数 C 语言实例 C 语言实例 查找字符在字符串中的起始位置(索引值从 开始). 实例 #include <stdio.h> int main( ...
- 掌握MySQL数据库这些优化技巧,事半功倍!
一个成熟的数据库架构并不是一开始设计就具备高可用.高伸缩等特性的,它是随着用户量的增加,基础架构才逐渐完善.这篇文章主要谈谈MySQL数据库在发展周期中所面临的问题及优化方案,暂且抛开前端应用不说,大 ...
- iOS开发 - RunLoop理解
RunLoop概念 运行循环,一个 run loop 就是一个事件处理的循环,用来不停的调度工作以及处理事件 作用 保持程序的持续运行 监听处理App中的各种事件(触摸事件,定时器事件,selecto ...
- 升级log4j到log4j2报错:cannot access org.apache.http.annotation.NotThreadSafe
问题与分析 今天把项目的log4j的依赖改成了log4j2的依赖后,发现使用Maven打包时报错如下: [ERROR] Failed to execute goal org.apache.maven. ...
- 【Codeforces1111D_CF1111D】Destroy the Colony(退背包_组合数学)
题目: Codeforces1111D 翻译: [已提交至洛谷CF1111D] 有一个恶棍的聚居地由几个排成一排的洞穴组成,每一个洞穴恰好住着一个恶棍. 每种聚居地的分配方案可以记作一个长为偶数的字符 ...
- Hive_Hive的管理_远程服务
远程服务启动方式 - 端口号10000 - 启动方式: #hive --service hiveserver & 以JDBC或ODBC的程序登陆到hive中操作数据时,必须选用远程服务启动方式 ...
- java threadLocal的初探
在网上找了半天,终于找到一篇靠谱的文章了. 文章地址:http://qifuguang.me/2015/09/02/[Java%E5%B9%B6%E5%8F%91%E5%8C%85%E5%AD%A6% ...
- split()分割字符串用法
<script type="text/javascript"> var str="How are you doing today?" documen ...
- top 进程管理
top 动态查看进程 前五行解释: 第一行参数说明: top - 07:06:19 当前时间 up 10 min, 系统运行时间,格式为时:分 1 user, 当前登录用户数 load av ...
- linux学习笔记汇总
linux 文件系统是采用级层树状的目录结构,采用"/"根目录的方式 目录结构: / 根目录 |---root: 存放root用户相关的文件 ...