hdu1754 I hate it线段树模板 区间最值查询
题目链接:这道题是线段树,树状数组最基础的问题
两种分类方式:按照更新对象和查询对象
单点更新,区间查询;
区间更新,单点查询;
按照整体维护的对象:
维护前缀和;
维护区间最值.
线段树模板代码
#include<iostream>
#include<string.h>
#include<stdio.h>
#include<math.h>
#include<algorithm>
using namespace std;
#define re(i,n) for(int i=0;i<n;i++)
typedef long long ll;
* 1e5 + ;
#define lson(x) x<<1,f,mid
#define rson(x) x<<1|1,mid+1,t
int n, m;
];
int x, y;
int ans;
void build(int r, int f, int t){
if (f == t){
scanf("%d", &tr[r]);
return;
}
;
build(lson(r)),build(rson(r));
tr[r] = max(tr[r << ], tr[r << | ]);
}
void query(int r, int f, int t){
if (f >= x&&t <= y){
ans = max(ans, tr[r]);
return;
}
;
if (x <= mid)query(lson(r));
if (y > mid)query(rson(r));
}
void update(int r, int f, int t){
if (f == t){
tr[r] = y;
return;
}
;
if (x<= mid)update(lson(r));
else update(rson(r));
tr[r] = max(tr[r << ], tr[r << | ]);
}
int main(){
//freopen("in.txt", "r", stdin);
while (cin >> n >> m){
build(, , n);
while (m--){
]; scanf("%s%d%d", op,&x,&y);
] == 'Q'){
if (x > y)swap(x, y);
ans = -;
query(, , n);
printf("%d\n", ans);
}
else{
update(, , n);
}
}
}
;
}
树状数组区间最值模板代码
#include<iostream>
#include<stdio.h>
#include<algorithm>
using namespace std;
*1e5+;
typedef long long ll;
#define re(i,n) for(int i=0;i<n;i++)
int a[maxn], c[maxn];
/*
a里面存放本值,c里面维护最值
*/
int n, q;
int lowbit(int x){
return x&-x;
}
void redo(int i){
c[i] = i;
; j < lowbit(i); j <<= )
if (a[c[i - j]]>a[c[i]])c[i] = c[i - j];
}
void init(){
; i <= n; i++){
redo(i);
}
}
void update(int x, int y){
bool big = y > a[x];
a[x] = y;
int i = x;
if (big){
/*
这个地方如果错写成a[c[i]]<y就会导致无法传递上去,因为一开始时就是a[c[i]]==y.
*/
while (i<=n&&a[c[i]] <= y)c[i] = x, i += lowbit(i);
}
else{
while (i<=n&&c[i] == x)redo(i), i += lowbit(i);
}
}
int query(int l, int r){
int ans = a[r];
while (l<=r){
while (r - lowbit(r) >= l){
ans = max(a[c[r]], ans);
r -= lowbit(r);
}
ans = max(a[r], ans);//根节点
r--;//向下走一步
}
return ans;
}
int main(){
freopen("in.txt", "r", stdin);
){
re(i, n)scanf(]);
init();
;
while (q--){
]; int x, y;
qi++;
scanf("%s%d%d", &op, &x, &y);
] == 'U'){
update(x, y);
}
else{
int ans = query(x, y);
printf("%d\n", ans);
}
}
}
;
}
复杂度是O(lgn*lgn).这道题数据好像有点弱,因为错误的代码也能过这道题.
zwk线段树模板
zwk线段树巧妙之处在于开区间写法,对于长度为N的线段,初始化时,共N+2片叶子:1和N+2空着,2~N+1才是本体,是实实在在的数据.张昆玮说了:如果有1023个数据,那就要有2048片叶子.空间要开到(N+2)*4才行.虽然费了点空间,但编程简洁了.
根节点(1号结点)和每层的第一个节点和最后一个节点是哨兵节点,永远都不会访问到它们.它们的存在只是方便编程.老子曰:将欲取之,必先予之.一言以蔽之,就是两端边界处的节点都是哨兵节点,这是开区间方便编程导致的.
那么闭区间写法行不行呢?对于长度为N的线段,1~N全部用上,查询时闭区间可以转换成开区间.如果有1023个数,那就只需要开辟1024片叶子的树状数组.开区间=闭区间-两个端点.若无论如何都更新两个端点,那就会导致更新的东西有点靠下,这对于结果并没什么影响.需要注意l==r的情形.
zwk开区间写法
#include<iostream>
#include<stdio.h>
#include<algorithm>
using namespace std;
* 1e5 + ;
typedef long long ll;
#define re(i,n) for(int i=0;i<n;i++)
];
int n, q, sz;
void update(int x, int v){
x += sz;
a[x] = v;
){
x >>= ;
a[x] = max(a[x << ], a[x << | ]);
}
}
int query(int l, int r){
l += sz - , r += sz + ;
;
){//当两人不是兄弟时
){
ans = max(ans, a[l ^ ]);
}
){
ans = max(ans, a[r ^ ]);
}
l >>= , r >>= ;
}
return ans;
}
int main(){
//freopen("in.txt", "r", stdin);
){
sz = ; )sz <<= ;
re(i, n){
scanf(]);
}
); i++)a[i + sz+] = ;
; i>; i--){
a[i] = max(a[i << ], a[i << | ]);
}
while (q--){
]; int x, y;
scanf("%s%d%d", op, &x, &y);
] == 'U'){
update(x, y);
}
else{
int ans = query(x, y);
printf("%d\n", ans);
}
}
}
;
}
zwk闭区间写法:下面的代码有bug.能过题.
#include<iostream>
#include<stdio.h>
#include<algorithm>
using namespace std;
* 1e5 + ;
typedef long long ll;
#define re(i,n) for(int i=0;i<n;i++)
];
int n, q, sz;
void update(int x, int v){
x += sz-;
a[x] = v;
){
x >>= ;
a[x] = max(a[x << ], a[x << | ]);
}
}
int query(int l, int r){
l += sz - , r += sz - ;
int ans =max(a[l],a[r]); //查询时先把两个端点给处理掉,就相当于开区间了.
/*
这个地方的不同决定了两种写法,我觉得没有必要开区间.闭区间预处理一下就很好.
这个地方如果l==r,就会产生死循环.但是我这么写,这道题却过了.*/
){//当两人不是兄弟时
){
ans = max(ans, a[l ^ ]);
}
){
ans = max(ans, a[r ^ ]);
}
l >>= , r >>= ;
}
return ans;
}
void init(){
sz = ; ;
re(i, n){
scanf("%d", &a[i + sz]);//在1~N之间存放数据
}
); i++)a[i + sz ] = ;
; i>; i--){
a[i] = max(a[i << ], a[i << | ]);
}
}
int main(){
freopen("in.txt", "r", stdin);
){
init();
while (q--){
]; int x, y;
scanf("%s%d%d", op, &x, &y);
] == 'U'){
update(x, y);
}
else{
int ans = query(x, y);
printf("%d\n", ans);
}
}
}
;
}
hdu1754 I hate it线段树模板 区间最值查询的更多相关文章
- hdu 1754 I Hate It (线段树求区间最值)
HDU1754 I Hate It Time Limit:3000MS Memory Limit:32768KB 64bit IO Format:%I64d & %I64u D ...
- xdoj-1324 (区间离散化-线段树求区间最值)
思想 : 1 优化:题意是覆盖点,将区间看成 (l,r)转化为( l-1,r) 覆盖区间 2 核心:dp[i] 覆盖从1到i区间的最小花费 dp[a[i].r]=min (dp[k])+a[i]s; ...
- HDU-1754 I Hate It(线段树,区间最大值)
很多学校流行一种比较的习惯.老师们很喜欢询问,从某某到某某当中,分数最高的是多少. 这让很多学生很反感. 不管你喜不喜欢,现在需要你做的是,就是按照老师的要求,写一个程序,模拟老师的询问.当然,老师 ...
- 2016年湖南省第十二届大学生计算机程序设计竞赛---Parenthesis(线段树求区间最值)
原题链接 http://acm.csu.edu.cn/OnlineJudge/problem.php?id=1809 Description Bobo has a balanced parenthes ...
- 滑动窗口(poj,线段树维护区间最值)
题目描述 现在有一堆数字共N个数字(N<=10^6),以及一个大小为k的窗口.现在这个从左边开始向右滑动,每次滑动一个单位,求出每次滑动后窗口中的最大值和最小值. 例如: The array i ...
- CodeForces 91B Queue (线段树,区间最值)
http://codeforces.com/problemset/problem/91/B B. Queue time limit per test: 2 seconds memory limit p ...
- POJ 2482 Stars in Your Window (线段树+扫描线+区间最值,思路太妙了)
该题和 黑书 P102 采矿 类似 参考链接:http://blog.csdn.net/shiqi_614/article/details/7819232http://blog.csdn.net/ts ...
- Balanced Lineup:线段树:区间最值 / RMQ
不要被线段树这个名字和其长长的代码吓到. D - Balanced Lineup Description For the daily milking, Farmer John's N cows (1 ...
- HDU6447 YJJ's Salesman-2018CCPC网络赛-线段树求区间最值+离散化+dp
目录 Catalog Solution: (有任何问题欢迎留言或私聊 && 欢迎交流讨论哦 Catalog Problem:Portal传送门 原题目描述在最下面. 1e5个点,问 ...
随机推荐
- PHP5.2至5.6的新增功能详解
截至目前(2014.2), PHP 的最新稳定版本是 PHP5.5, 但有差不多一半的用户仍在使用已经不在维护 [注] 的 PHP5.2, 其余的一半用户在使用 PHP5.3 [注]. 因为 PHP ...
- CentOS 6.3下rsync服务器的安装与配置
一.rsync 简介 Rsync(remote synchronize)是一个远程数据同步工具,可通过LAN/WAN快速同步多台主机间的文件,也可以使用 Rsync 同步本地硬盘中的不同目录. Rsy ...
- 初涉Linux ----------> Ubuntu15.04的安装与美化
“你玩 Linux 吗?” “什么 Linux ?” “你连 Linux 都不知道?还说是学计算机的呢...” “干嘛要用 Linux 啊?windows多好,Linux?没兴趣” 一. 前言 ...
- Hadoop Kernel tunning
/etc/security/limits.conf @ochadoop soft nofile 102642 @ochadoop hard nofile 102642 @ochadoop soft n ...
- 准备使用 Office 365 中国版--域名
无论是购买还是试用Office 365,每个Office 365用户都会有一个唯一的ID.在中国版的Office 365中,这个ID是一个以“.partner.onmschina.cn”结尾的域名.既 ...
- label的for属性
一.使用介绍 <label>专为input元素服务,为其定义标记. for属性规定label与哪个表单元素绑定 label和表单控件绑定方式又两种: 1.将表单控件作为label的内容,这 ...
- SQL--实现分页查询
在查询数据中,对于某些数据量过大,为了减少页面上单页的加载时间,我们常常会选择分页查询,分页查询有很多方法,下面主要介绍两种分页方法. 一. 通过主键来实现分页: 1.数据库背景. P ...
- nginx集群报错“upstream”directive is not allow here 错误
nginx集群报错“upstream”directive is not allow here 错误 搭建了一个服务器, 采用的是nginx + apache(多个) + php + mysql(两个) ...
- PL/SQL Transaction Control
PL/SQL 基础 ( 下 ) 1. PL/SQL中的 SQL语句 - END语句与COMMIT等内容,没有任何关系. - PL/SQL does not directly support dat ...
- VS2010 生成Xml格式的注释文档
项目, 属性, build, 勾选xml document file, 重新build, 即可生成xml注释文件, 然后还得找工具软件(看到anytao推荐SandCastle) 生成更易读的帮助文档 ...