hihocoder 1077线段树

http://hihocoder.com/problemset/problem/1077

#include <bits/stdc++.h>
using namespace std;
#define ll long long
#define co(x) cout << (x) << endl
#define ci(x) cin >> (x)
#define sd(x) scanf("%d",&x)
#define sf(x) scanf("%lf",&x)
#define pc(x) printf("%c",x)
#define pd(x) printf("%d",x)
#define gcd(x,y) __gcd(x,y)
#define w(x) while(x)
#define fo(i,j,k) for(int (i) = (j); (i) < (k); (i)++)
#define en cout << endl;
#define INF 2147483645
#define Maxn 1000010

struct Node{
    int left,right;
    int val;
}A[Maxn<<2];

void Build(int i,int left,int right){
    A[i].left = left;
    A[i].right = right;
    if(left == right){//如果该结点是根节点就赋值
        ci(A[i].val);
        return ;
    }
    int mid = (left+right) >> 1;
    Build( i<<1,left,mid );//向两边递归
    Build(i<<1|1,mid+1,right );
    A[i].val = min( A[i<<1].val,A[i<<1|1].val );
    //取较小的值(其他类比)
}

void update(int i,int p,int val){
    if( A[i].left == A[i].right ){//找到根结点
        A[i].val = val;//修改
        return ;
    }
    if( p <= A[i<<1].right ){//如果结点P在比A[i]的右儿子的右区间小(在右儿子的区间内)
        update( i<<1,p,val );//向右儿子的左区间更新节点
    }
    if( p >= A[i<<1|1].left ){//如果比左儿子的左区间大,则向右更新节点
        update(i<<1|1,p,val);
    }//向两边同时递归更新节点的值,保证每个被影响的值都被更新
    A[i].val = min( A[i<<1].val,A[i<<1|1].val );
}
// 如果该行描述一次商品的重量的更改,则接下来为两个整数Pi,Wi,
// 表示位置编号为Pi的商品的重量变更为Wi

int query(int i,int left,int right){ //left为查询的区间
    if( A[i].left >= left && A[i].right <= right ){//在查询区间内,返回该节点的值??
        return A[i].val;
    }
    int a = INF,b = INF;
    if( left <= A[i<<1].right ){ //如果左范围在A[i]的右儿子的左边,就递归向左边查询
        a = query(i << 1,left,right);
    }
    if( right >= A[i<<1|1].left ){//如果右范围在A[i]的左儿子的右边,就递归向右查询
        b = query(i<<1|1,left,right);
    }
    return min(a,b);//回朔返回左右儿子的较小值
}

int main(){
    int N,M,a,b,c;
    while( sd(N)!=EOF ){
        Build(1,1,N);
        ci(M);
        fo(i,0,M){
            scanf("%d %d %d",&a,&b,&c);
            if(a){
                update(1,b,c);
            }else{
                printf("%d\n",query(1,b,c));
            }
        }
    }
}
时间: 2024-10-13 16:16:23

hihocoder 1077线段树的相关文章

hihocoder 1080 线段树(区间更新)

题目链接:http://hihocoder.com/problemset/problem/1080 , 两种操作的线段树(区间更新). 这道题前一段时间一直卡着我,当时也是基础不扎实做不出来,今天又想了想其实还是比较简单的,也只能怪自己太弱了. 这道题坑就坑在是有两个操作:set和add,所以lazy标记数组就需要两个,但是有一点要考虑的是一个区间上set与add的先后关系——如果对于一个区间set和add标记同时存在,那么应该如果处理:一种情况set在add之前,那么就按照正常顺序来就可以了:

hihoCoder #1078 : 线段树的区间修改(线段树区间更新板子题)

#1078 : 线段树的区间修改 时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描述 对于小Ho表现出的对线段树的理解,小Hi表示挺满意的,但是满意就够了么?于是小Hi将问题改了改,又出给了小Ho: 假设货架上从左到右摆放了N种商品,并且依次标号为1到N,其中标号为i的商品的价格为Pi.小Hi的每次操作分为两种可能,第一种是修改价格——小Hi给出一段区间[L, R]和一个新的价格NewP,所有标号在这段区间中的商品的价格都变成NewP.第二种操作是询问——小Hi给出一段

hihoCoder#1079(线段树+坐标离散化)

时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描述 小Hi和小Ho在回国之后,重新过起了朝7晚5的学生生活,当然了,他们还是在一直学习着各种算法~ 这天小Hi和小Ho所在的学校举办社团文化节,各大社团都在宣传栏上贴起了海报,但是贴来贴去,有些海报就会被其他社团的海报所遮挡住.看到这个场景,小Hi便产生了这样的一个疑问——最后到底能有几张海报还能被看见呢? 于是小Ho肩负起了解决这个问题的责任:因为宣传栏和海报的高度都是一样的,所以宣传栏可以被视作长度为L的一段区间,且有

hihocoder1070、1077线段树更新点查询区间

题目链接:http://hihocoder.com/problemset/problem/1070 http://hihocoder.com/problemset/problem/1077 我的代码: 1 #include <cstdio> 2 #include <algorithm> 3 4 using namespace std; 5 6 #define MAXN 1000005 7 8 int w[MAXN]; 9 10 struct segNode 11 { 12 int

hihoCoder #1078 : 线段树的区间修改

思路: 1 #include <bits/stdc++.h> 2 using namespace std; 3 const int N=100010; 4 int n, q, op, l, r, flag, sma; 5 int w[N]; 6 int ans[N]; 7 8 void cal(int t)//从t开始计算,ans[i]表示1~i的和 9 { 10 for(int i=t; i<=n; i++) 11 ans[i]=ans[i-1]+w[i]; 12 } 13 14 in

hihoCoder week20 线段树的区间修改

区间修改 区间查询 最后一场比赛前的无可救药的热身 #include <bits/stdc++.h> using namespace std; #define mid ((l+r)/2) #define ls (rt<<1) #define rs (rt<<1|1) const int N = 2e5+10; int n, s[N], tr[N], lz[N]; void build(int rt,int l,int r) { if(l ==r) { tr[rt] =

hihoCoder #1077 RMQ问题再临-线段树

#1077 : RMQ问题再临-线段树 Time Limit:10000ms Case Time Limit:1000ms Memory Limit:256MB 描述 上回说到:小Hi给小Ho出了这样一道问题:假设整个货架上从左到右摆放了N种商品,并且依次标号为1到N,每次小Hi都给出一段区间[L, R],小Ho要做的是选出标号在这个区间内的所有商品重量最轻的一种,并且告诉小Hi这个商品的重量.但是在这个过程中,可能会因为其他人的各种行为,对某些位置上的商品的重量产生改变(如更换了其他种类的商品

Hihocoder #1077 : RMQ问题再临-线段树(线段树:结构体建树+更新叶子往上+查询+巧妙使用father[]+线段树数组要开大4倍 *【模板】)

#1077 : RMQ问题再临-线段树 时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描述 上回说到:小Hi给小Ho出了这样一道问题:假设整个货架上从左到右摆放了N种商品,并且依次标号为1到N,每次小Hi都给出一段区间[L, R],小Ho要做的是选出标号在这个区间内的所有商品重量最轻的一种,并且告诉小Hi这个商品的重量.但是在这个过程中,可能会因为其他人的各种行为,对 某些位置上的商品的重量产生改变(如更换了其他种类的商品). 小Ho提出了两种非常简单的方法,但是都不能

hihoCoder 1080 : 更为复杂的买卖房屋姿势 线段树区间更新

#1080 : 更为复杂的买卖房屋姿势 时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描述 小Hi和小Ho都是游戏迷,“模拟都市”是他们非常喜欢的一个游戏,在这个游戏里面他们可以化身上帝模式,买卖房产. 在这个游戏里,会不断的发生如下两种事件:一种是房屋自发的涨价或者降价,而另一种是政府有关部门针对房价的硬性调控.房价的变化自然影响到小Hi和小Ho的决策,所以他们希望能够知道任意时刻某个街道中所有房屋的房价总和是多少——但是很不幸的,游戏本身并不提供这样的计算.不过这难