今天学习了线段树的三个基本操作 建树 更新 查找

先理解下什么是线段树
就这个题目而言 如果我们用普通的数组去存放 然后依次遍历访问的话 时间太多了
线段树利用了二分的思想 把数据以段的形式进行储存 这样在访问的时候 时间复杂度就下来了

上图就是线段树的一个简单的模型

创建线段树(初始化)】:

由于线段树是用二叉树结构储存的,而且是近乎完全二叉树的,所以在这里我使用了数组来代替链表上图中区间上面的红色数字表示了结构体数组中对应的下标。

在完全二叉树中假如一个结点的序号(数组下标)为 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的更多相关文章

  1. [线段树]HDU-1754板子题入门ver

    HDU-1754 线段树数组请开到四倍 众所周知数组开小会导致re tle wa等一系列问题orz 板子就是板子,数组从零开始或是从一开始都没什么问题,就是2*root+1还是2*root+2的问题. ...

  2. 线段树---HDU1754 I hate it

    这个题也是线段树的基础题,有了上一个题的基础,在做这个题就显得比较轻松了,大体都是一样的,那个是求和,这个改成求最大值,基本上思路差不多,下面是代码的实现 #include <cstdio> ...

  3. 线段树 HDU-1754 I Hate It

    附上原题链接:http://acm.hdu.edu.cn/showproblem.php?pid=1754 Problem Description 很多学校流行一种比较的习惯.老师们很喜欢询问,从某某 ...

  4. 线段树hdu1754

    #include<iostream>#include<stdio.h>using namespace std;const int maxa=200005;int val[max ...

  5. 几道简单的线段树入门题 POJ3264&&POJ3468&&POJ2777

    Balanced Lineup Time Limit: 5000MS   Memory Limit: 65536K Total Submissions: 40687   Accepted: 19137 ...

  6. I Hate It(hdu1754)(线段树区间最大值)

    I Hate It hdu1754 Time Limit: 9000/3000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others ...

  7. HDU1754 && HDU1166 线段树模板题

    HDU1754 题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1754 题目分析:对于给出的一个很长的区间,对其进行单点更新值和区间求最大值的操作,由于 ...

  8. HDU1166(线段树 +更新单点,求区间总和)、HDU1754(线段树 + 更新单点,求区间最大值)

    线段树简单应用 先附上几张图便与理解,大佬文章传送门1.传送门2 HDU1166:题目描述 线段树 +更新单点,求区间总和 代码如下(递归版) #include<iostream> #in ...

  9. hdu1754 I hate it线段树模板 区间最值查询

    题目链接:这道题是线段树,树状数组最基础的问题 两种分类方式:按照更新对象和查询对象 单点更新,区间查询; 区间更新,单点查询; 按照整体维护的对象: 维护前缀和; 维护区间最值. 线段树模板代码 # ...

随机推荐

  1. Alpha项目冲刺! Day5-产出

    各个成员今日完成的任务 林恩:任务分工,博客撰写,了解安卓环境搭建 杨长元:安卓本地数据库基本建立 李震:改了图标和背景 胡彤:完善服务端 寇永明:研究测试代码 王浩:研究测试代码 李杰:研究测试代码 ...

  2. 到达型01背包---P1877 [HAOI2012]音量调节

    P1877 [HAOI2012]音量调节 题解 solution 1 普通dfs  60pt dfs 暴搜,pos 记录当前到了第几首歌,level 记录当前的音量 一个小剪枝 由于每换一首歌都要调节 ...

  3. java 接口和抽象类的一个最大的区别

    写在前面,下面是在百度百科上看到的,之前就看过,这次再看感觉有更深的体会,真的是这样,每一个脚印都会留下痕迹 java接口和java抽象类有太多相似的地方,又有太多特别的地方,这里说下两者之间的一个最 ...

  4. http-proxy-middleware及express实现反向代理

    $ npm install --save-dev http-proxy-middleware npm install express // 引用依赖 var express = require('ex ...

  5. MySQL数据库之慢查询日志

    一.开启慢查询日志 通过show global variables like '%slow%' #查看MySQL慢查询日志是否开启 [root@mysqlmaster01 ~]# mysql --lo ...

  6. [Feature] Feature selection - Embedded topic

    基于惩罚项的特征选择法 一.直接对特征筛选 Ref: 1.13.4. 使用SelectFromModel选择特征(Feature selection using SelectFromModel) 通过 ...

  7. 123467123456#1#-----com.twoapp.DaDiShuGame01--前拼后广--现实打地鼠游戏jiemei

    com.twoapp.DaDiShuGame01--前拼后广--现实打地鼠游戏jiemei

  8. 123457---小小数学家--com.twoapp.xiaoxiaoshuxuejia

    小小数学家--com.twoapp.xiaoxiaoshuxuejia

  9. 利用python批量修改word文件名的方法示例

    利用python批量修改word文件名的方法示例 最近不小心把硬盘给格式化了,由于当时的文件没有备份,所以一下所有的文件都没有了,于是只能采取补救措施,用文件恢复软件恢复了一部分的数据出来,但是恢复完 ...

  10. mysl创建用户+授权+增、删、改查

    1.mysql的root用户无法给普通用户授权问题处理 update mysql.user set Grant_priv='Y' where User='root' and Host='%': flu ...