入手线段树 hdu1754
今天学习了线段树的三个基本操作 建树 更新 查找 先理解下什么是线段树
就这个题目而言 如果我们用普通的数组去存放 然后依次遍历访问的话 时间太多了
线段树利用了二分的思想 把数据以段的形式进行储存 这样在访问的时候 时间复杂度就下来了


上图就是线段树的一个简单的模型
创建线段树(初始化)】:
由于线段树是用二叉树结构储存的,而且是近乎完全二叉树的,所以在这里我使用了数组来代替链表上图中区间上面的红色数字表示了结构体数组中对应的下标。
在完全二叉树中假如一个结点的序号(数组下标)为 I ,那么 (二叉树基本关系)
I 的父亲为 I/2,
I 的另一个兄弟为 I/2*2 或 I/2*2+1
I 的两个孩子为 I*2 (左) I*2+1(右)
这段来自其他同学的博客 这个是建树的基础 我们用标号来链接每个叶节点
struct node
{
int maxx;
int l,r;
}stu[maxn*4]; 这个结构体就是每一个线段树的一个叶 其中有左右节点以及本题需要的max值
先说说建树操作把
int mid=(l+r)/2;
stu[i].l=l;
stu[i].r=r;
既然利用到了二分的思想 那么 这几步是少不了的
void buildtree(int i,int l,int r)//建树
{
int mid=(l+r)/2;
stu[i].l=l;
stu[i].r=r;
if(l==r)
{
stu[i].maxx=a[l];
return;
}
buildtree(i*2,l,mid);//构建左子点
buildtree(i*2+1,mid+1,r);//构建右子点
stu[i].maxx=max(stu[i*2].maxx,stu[i*2+1].maxx);//在递归结束回溯的时候 把最大值返回
}
再说说数据的更新
更新的话 先得二分找到目标位置 然后替换 最后还得依次更新最大值的数据
void updata(int i)//更新
{
int l=stu[i].l;
int r=stu[i].r;
int mid=(l+r)/2;//二分咯
if(l==r&&r==x)//x为目标id 当左右节点相同的时候 就是找到这个数的时候
{
stu[i].maxx=y;
return;
}
if(l<=x&&x<=mid) updata(i*2);//向左找
else updata(i*2+1);//向右找
stu[i].maxx=max(stu[i*2].maxx,stu[i*2+1].maxx);//回溯的时候 更新max的数据
}
最后就是询问过程了
找最大值的过程中 要把要求的区间分段 即分割成好几个小段 然后放在建好的书里面找
int que(int a,int b,int i)//查找
{
int l=stu[i].l;
int r=stu[i].r;
int mid=(l+r)/2;
if(l==a&&r==b) return stu[i].maxx;
if(b<=mid) return que(a,b,i*2);//全在右边
else if(a>mid) return que(a,b,i*2+1);//全在左边
else return max(que(a,mid,i*2),que(mid+1,b,i*2+1));//需要分割的话 取分开的最大
}
上完整代码
#include<cstdio>
#include<iostream>
#include<string.h>
using namespace std;
#define maxn 200010
struct node
{
int maxx;
int l,r;
}stu[maxn*4];
int a[maxn];
int n,m,x,y;
void buildtree(int i,int l,int r)//建树
{
int mid=(l+r)/2;
stu[i].l=l;
stu[i].r=r;
if(l==r)
{
stu[i].maxx=a[l];
return;
}
buildtree(i*2,l,mid);
buildtree(i*2+1,mid+1,r);
stu[i].maxx=max(stu[i*2].maxx,stu[i*2+1].maxx);
}
void updata(int i)//更新
{
int l=stu[i].l;
int r=stu[i].r;
int mid=(l+r)/2;
if(l==r&&r==x)
{
stu[i].maxx=y;
return;
}
if(l<=x&&x<=mid) updata(i*2);
else updata(i*2+1);
stu[i].maxx=max(stu[i*2].maxx,stu[i*2+1].maxx);
}
int que(int a,int b,int i)//查找
{
int l=stu[i].l;
int r=stu[i].r;
int mid=(l+r)/2;
if(l==a&&r==b) return stu[i].maxx;
if(b<=mid) return que(a,b,i*2);
else if(a>mid) return que(a,b,i*2+1);
else return max(que(a,mid,i*2),que(mid+1,b,i*2+1));
}
int main()
{
char z;
cin.sync_with_stdio(false);
while(cin>>n>>m)
{
for(int i=1;i<=n;i++) cin>>a[i];
buildtree(1,1,n);
while(m--)
{
cin>>z>>x>>y;
if(z=='U')
{
updata(1);
}
else
{ cout<<que(x,y,1)<<endl;
}
}
}
return 0;
}
入手线段树 hdu1754的更多相关文章
- [线段树]HDU-1754板子题入门ver
HDU-1754 线段树数组请开到四倍 众所周知数组开小会导致re tle wa等一系列问题orz 板子就是板子,数组从零开始或是从一开始都没什么问题,就是2*root+1还是2*root+2的问题. ...
- 线段树---HDU1754 I hate it
这个题也是线段树的基础题,有了上一个题的基础,在做这个题就显得比较轻松了,大体都是一样的,那个是求和,这个改成求最大值,基本上思路差不多,下面是代码的实现 #include <cstdio> ...
- 线段树 HDU-1754 I Hate It
附上原题链接:http://acm.hdu.edu.cn/showproblem.php?pid=1754 Problem Description 很多学校流行一种比较的习惯.老师们很喜欢询问,从某某 ...
- 线段树hdu1754
#include<iostream>#include<stdio.h>using namespace std;const int maxa=200005;int val[max ...
- 几道简单的线段树入门题 POJ3264&&POJ3468&&POJ2777
Balanced Lineup Time Limit: 5000MS Memory Limit: 65536K Total Submissions: 40687 Accepted: 19137 ...
- I Hate It(hdu1754)(线段树区间最大值)
I Hate It hdu1754 Time Limit: 9000/3000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others ...
- HDU1754 && HDU1166 线段树模板题
HDU1754 题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1754 题目分析:对于给出的一个很长的区间,对其进行单点更新值和区间求最大值的操作,由于 ...
- HDU1166(线段树 +更新单点,求区间总和)、HDU1754(线段树 + 更新单点,求区间最大值)
线段树简单应用 先附上几张图便与理解,大佬文章传送门1.传送门2 HDU1166:题目描述 线段树 +更新单点,求区间总和 代码如下(递归版) #include<iostream> #in ...
- hdu1754 I hate it线段树模板 区间最值查询
题目链接:这道题是线段树,树状数组最基础的问题 两种分类方式:按照更新对象和查询对象 单点更新,区间查询; 区间更新,单点查询; 按照整体维护的对象: 维护前缀和; 维护区间最值. 线段树模板代码 # ...
随机推荐
- golang通过ssh实现远程文件传输
使用ssh远程操作文件, 主要是创建ssh, 直接上代码 import ( "fmt" "github.com/pkg/sftp" "golang.o ...
- 骑行川藏--新都桥&塔公草原
新都桥 塔公草原 新都桥,位于四川省甘孜藏族自治州康定市西部地区,距市区81公里: 别名:东俄罗,一个镇名.海拔约3300米,没有突出的标志性景观,沿线有10余公里被称为“摄影家走廊”. 神奇光线,无 ...
- BitmapFactory之Options
package com.loaderman.customviewdemo; import android.app.Activity; import android.graphics.Bitmap; i ...
- BitmapShader填充图形
package com.loaderman.customviewdemo; import android.content.Context; import android.graphics.*; imp ...
- MySQL中表的复制以及大型数据表的备份教程
MySQL中表的复制以及大型数据表的备份教程 这篇文章主要介绍了MySQL中表的复制以及大型数据表的备份教程,其中大表备份是采用添加触发器增量备份的方法,需要的朋友可以参考下 表复制 mysq ...
- Windows下安装ActiveMQ
到官网(http://activemq.apache.org/download-archives.html)下载最新发布的压缩包(我下的是5.15.9)到本地后解压(我解压到D盘Dev目录下)即可.进 ...
- 123457123456#0#-----com.twoapp.mathGame13--前拼后广--13种数学方法jiemei
com.twoapp.mathGame13--前拼后广--13种数学方法jiemei
- 123457123456#0#-----com.twoapp.jingPinYinYu01----儿童学英语jiemei
com.twoapp.jingPinYinYu01----儿童学英语jiemei
- 第二十二章 集成验证码——《跟我学Shiro》
目录贴:跟我学Shiro目录贴 在做用户登录功能时,很多时候都需要验证码支持,验证码的目的是为了防止机器人模拟真实用户登录而恶意访问,如暴力破解用户密码/恶意评论等.目前也有一些验证码比较简单,通过一 ...
- 实现一个Promise
实现一个Promise promise特点 一个promise的当前状态只能是pending.fulfilled和rejected三种之一.状态改变只能是pending到fulfilled或者pend ...