线段树点修改模板(HDU1754)

很多学校流行一种比较的习惯。老师们很喜欢询问,从某某到某某当中,分数最高的是多少。
这让很多学生很反感。

不管你喜不喜欢,现在需要你做的是,就是按照老师的要求,写一个程序,模拟老师的询问。当然,老师有时候需要更新某位同学的成绩。

Input

本题目包含多组测试,请处理到文件结束。
在每个测试的第一行,有两个正整数 N 和 M ( 0<N<=200000,0<M<5000 ),分别代表学生的数目和操作的数目。
学生ID编号分别从1编到N。
第二行包含N个整数,代表这N个学生的初始成绩,其中第i个数代表ID为i的学生的成绩。
接下来有M行。每一行有一个字符 C (只取‘Q‘或‘U‘) ,和两个正整数A,B。
当C为‘Q‘的时候,表示这是一条询问操作,它询问ID从A到B(包括A,B)的学生当中,成绩最高的是多少。
当C为‘U‘的时候,表示这是一条更新操作,要求把ID为A的学生的成绩更改为B。

Output

对于每一次询问操作,在一行里面输出最高成绩。

Sample Input

5 6
1 2 3 4 5
Q 1 5
U 3 6
Q 3 4
Q 4 5
U 2 9
Q 1 5

Sample Output

5
6
5
9

Hint

Huge input,the C function scanf() will work better than cin

Author

linle

*****************************************************************************************************************************************************

需要注意的是scanf读取时会读入‘\n‘,在代码中提供了解决方法。

 1 #include<stdio.h>
 2 #define MAXSIZE 200000
 3 int val[MAXSIZE+1];
 4 int max(int x,int y){
 5     return x>y?x:y;
 6 }
 7 struct node{
 8     int Max,left,right;
 9 } tree[MAXSIZE*3];
10
11 int creat(int root,int left,int right){
12     tree[root].left=left;
13     tree[root].right=right;
14     if(left == right)
15         return tree[root].Max=val[left];
16     int a,b,mid=(left+right)>>1;
17     a=creat(root<<1,left,mid);
18     b=creat(root<<1|1,mid+1,right);
19     return tree[root].Max=max(a,b);
20 }
21 int calculate(int root,int left,int right){
22     if(tree[root].left>right||tree[root].right<left)
23         return 0;
24     if(left<=tree[root].left&&tree[root].right<=right)
25         return tree[root].Max;
26     int a,b;
27     a=calculate(root<<1,left,right);
28     b=calculate(root<<1|1,left,right);
29     return max(a,b);
30 }
31 int update(int root,int pos,int val){
32     if(pos<tree[root].left||tree[root].right<pos)
33         return tree[root].Max;
34     if(tree[root].left==pos&&tree[root].right==pos)
35         return tree[root].Max=val;
36     int a,b;
37     a=update(root<<1,pos,val);
38     b=update(root<<1|1,pos,val);
39     return tree[root].Max=max(a,b);
40 }
41 int main()
42 {
43     int n,m;
44     while(scanf("%d%d",&n,&m)!=EOF){
45         for(int i=1;i<=n;i++){
46             scanf("%d",&val[i]);
47         }
48         creat(1,1,n);
49         for(int i=0;i<m;i++){
50             char op;
51             int a,b;
52             scanf("\n%c%d%d",&op,&a,&b);
53             if(op==‘Q‘)
54                 printf("%d\n",calculate(1,a,b));
55             else
56                 update(1,a,b);
57         }
58     }
59 }
时间: 2024-10-12 16:27:38

线段树点修改模板(HDU1754)的相关文章

HDU 1166 敌兵布阵 (我的树状数组加线段树点修改模板)

思路:本题因为是点修改,所以我们可以用线段树或者是树状数组了.线段树的基本操作我在我的代码中会具体体现,关键是要理解下面这幅图,具体的思想大家可以去看看其他的资料 线段树AC代码: #include<iostream> #include<cstdio> #include<cmath> #include<cstring> #include<algorithm> using namespace std; #define N 50005 int num

线段树区间修改模板

本来打算把大白书第三章一口气攻下来的,但是这个线段树也是卡了好久. 不敢过题太快,怕自己走马观花到头来结果什么都不会. 可也不能再拖了,在做题中也许有更多的体会. 模板一:1 L R v 表示区间[L, R]所有元素都加上v2 L R   表示查询区间[L, R]的sum, min, maxsumv[o]的定义为:如果只执行节点o及其子孙节点的中的add操作,节点o对应区间中所有数之和 1 //线段树区间修改 2 //1 L R v 表示区间[L, R]所有元素都加上v 3 //2 L R 表示

P1198最大数——线段树点修改&amp;&amp;模板题

题目 题目链接 大意:维护一个数列,有两种操作: 查询操作Q  L:查询当前数列中末尾L个数中的最大的数 插入操作A  n:将n加上t再对D取模,将所得值插入数列末尾 解决方案 由题意知,只有两种操作:单点修改.区间查询 1 #include<bits/stdc++.h> 2 using namespace std; 3 4 typedef long long ll; 5 const ll INF = (ll)1 << 60; 6 const int maxn = 200000 +

【模板】线段树区间修改

区间修改: 区间修改过程类似于区间询问,例如将[ul, ur]内的所有元素都加上v,则进行如下操作: 当当前区间被区间[ul, ur]所包含时, 当前的节点值加上区间长度(r - l  + 1)乘以v 对当前节点的lazy-tag加上v,修改结束 否则,将当前节点的lazy-tag下传,分别修改左孩子和右孩子(一定条件下),然后更新此节点的值 lazy-tag下传: 如果当前节点没有lazy-tag,直接return 否则:1. 将左孩子右孩子分别加上lazy-tag * 区间长度的值  2.

hdu1698 Just a Hook(线段树+区间修改+区间查询+模板)

Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submission(s): 54923    Accepted Submission(s): 25566 Problem Description In the game of DotA, Pudge’s meat hook is actually the most horrible thing for most of

线段树区间修改 P3372 【模板】线段树 1

题目描述 如题,已知一个数列,你需要进行下面两种操作: 1.将某区间每一个数加上x 2.求出某区间每一个数的和 输入输出格式 输入格式: 第一行包含两个整数N.M,分别表示该数列数字的个数和操作的总个数. 第二行包含N个用空格分隔的整数,其中第i个数字表示数列第i项的初始值. 接下来M行每行包含3或4个整数,表示一个操作,具体如下: 操作1: 格式:1 x y k 含义:将区间[x,y]内每个数加上k 操作2: 格式:2 x y 含义:输出区间[x,y]内每个数的和 输出格式: 输出包含若干行整

【模板】线段树-单点修改,区间查询

容易理解但是难打(又长又难调)------仅代表个人观点 (能别打就别打) 线段树是什么? 大概长这样?(表示区间1到6) 线段树是一颗二叉树,是通过二分思想建立的一颗表示区间关系的树形结构.(总之记住它很好用就对了) 怎样建一颗线段树 大概思路: 二分+递归 没什么好讲的,具体看代码吧.. //建树 struct node { int a,b; }tree[100001]; void make_tree(int p,int x,int y)//p为当前节点编号,x,y为区间的左右端点 { tr

线段树lazytag优化模板

线段树嘛...很基本的一种数据结构啦 lazytag优化能不错的提高效率 1 #include<bits/stdc++.h> 2 using namespace std; 3 const int MAX=1000000; 4 struct pr { 5 int sum; 6 int lazy; 7 int left,right; 8 }tr[MAX+10]; 9 int n; 10 inline int ll(int k) {return 2*k;} 11 inline int rr(int

poj 2777 Count Color(线段树区间修改)

题目链接:http://poj.org/problem?id=2777 题目意思:就是问你在询问的区间里有几种不同的颜色 思路:这题和一般的区间修改差不多,但是唯一不同的就是我们要怎么计算有种颜色,所以这时候我们就需要把延时标记赋予不同的意义,当某段区间有多种颜色时就赋值为-1,当为一种颜色时就把它赋值为这个颜色的号数.这儿我们要怎么统计询问区间不同的颜色数叻,为了不重复计算同一种颜色,那么我们就需要用一个数组来标记计算过的颜色,当我们下次遇到时就不需要再次计算了.... 代码核心处就在计数那儿