HDOJ1754 I Hate It 【线段树】

I Hate It

Time Limit: 9000/3000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)

Total Submission(s): 35417    Accepted Submission(s): 13958

Problem Description

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

这让很多学生很反感。

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

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

这题验证了宏定义要比自定义函数慢。但不知道为啥,然后问了下老师,答复是“函数调用参数是计算好的,只计算一次;宏定义没有计算,运行时每次都要计算。”

Run ID Submit Time Judge Status Pro.ID Exe.Time Exe.Memory Code Len. Language Author
10955721 2014-07-07 11:20:55 Accepted 1754 1109MS 7156K 1513 B G++ 长木
10955685 2014-07-07 11:18:05 Time Limit Exceeded 1754 3000MS 7144K 1491 B G++ 长木

AC:

#include <stdio.h>
#define maxn 200002

struct Node{
	int left, right;
	int max;
} tree[maxn << 2];
int stu[maxn];

int MAX(int a, int b)
{
	return a > b ? a : b;
}

void build(int left, int right, int rt)
{
	tree[rt].left = left; tree[rt].right = right;
	if(left == right){
		tree[rt].max = stu[left]; return;
	}

	int mid = (left + right) >> 1;
	build(left, mid, rt << 1);
	build(mid + 1, right, rt << 1 | 1);

	tree[rt].max = MAX(tree[rt<<1].max, tree[rt<<1|1].max);
}

void update(int pos, int val, int rt)
{
	if(tree[rt].left == tree[rt].right){
		tree[rt].max = val; return;
	}

	int mid = (tree[rt].left + tree[rt].right) >> 1;
	if(pos <= mid) update(pos, val, rt << 1);
	else update(pos, val, rt << 1 | 1);

	tree[rt].max = MAX(tree[rt<<1].max, tree[rt<<1|1].max);
}

int query(int left, int right, int rt)
{
	if(tree[rt].left == left && right == tree[rt].right){
		return tree[rt].max;
	}

	int mid = (tree[rt].left + tree[rt].right) >> 1;
	if(right <= mid) return query(left, right, rt << 1);
	else if(left > mid)
		return query(left, right, rt << 1 | 1);
	return MAX(query(left, mid, rt << 1), query(mid+1, right, rt<<1|1));
}

int main()
{
	int N, M, i, m, n;
	char com[2];
	while(scanf("%d%d", &N, &M) == 2){
		for(i = 1; i <= N; ++i)
			scanf("%d", stu + i);
		build(1, N, 1);

		while(M--){
			scanf("%s%d%d", com, &m, &n);

			if(com[0] == 'U') update(m, n, 1);
			else printf("%d\n", query(m, n, 1));
		}
	}
	return 0;
}

TLE::

#include <stdio.h>
#define MAX(a, b) a > b ? a : b
#define maxn 200002

struct Node{
    int lson, rson;
    int max;
} tree[maxn << 2];
int stu[maxn];

void build(int left, int right, int rt)
{
    tree[rt].lson = left; tree[rt].rson = right;
    if(left == right){
        tree[rt].max = stu[left]; return;
    }

    int mid = (left + right) >> 1;
    build(left, mid, rt << 1);
    build(mid + 1, right, rt << 1 | 1);

    tree[rt].max = MAX(tree[rt<<1].max, tree[rt<<1|1].max);
}

void update(int pos, int val, int rt)
{
    if(tree[rt].lson == tree[rt].rson){
        tree[rt].max = val; return;
    }

    int mid = (tree[rt].lson + tree[rt].rson) >> 1;
    if(pos <= mid) update(pos, val, rt << 1);
    else update(pos, val, rt << 1 | 1);

    tree[rt].max = MAX(tree[rt<<1].max, tree[rt<<1|1].max);
}

int query(int left, int right, int rt)
{
    if(tree[rt].lson == left && right == tree[rt].rson){
        return tree[rt].max;
    }

    int mid = (tree[rt].lson + tree[rt].rson) >> 1;
    if(right <= mid) return query(left, right, rt << 1);
    else if(left > mid)
        return query(left, right, rt << 1 | 1);
    return MAX(query(left, mid, rt << 1), query(mid+1, right, rt<<1|1));
}

int main()
{
    int N, M, i, m, n;
    char com[2];
    while(scanf("%d%d", &N, &M) == 2){
        for(i = 1; i <= N; ++i)
            scanf("%d", stu + i);
        build(1, N, 1);

        while(M--){
            scanf("%s%d%d", com, &m, &n);

            if(com[0] == 'U') update(m, n, 1);
            else printf("%d\n", query(m, n, 1));
        }
    }
    return 0;
}

HDOJ1754 I Hate It 【线段树】

时间: 2024-08-05 11:05:35

HDOJ1754 I Hate It 【线段树】的相关文章

线段树-点修改-hdoj-1754

I Hate It Problem Description 很多学校流行一种比较的习惯.老师们很喜欢询问,从某某到某某当中,分数最高的是多少. 这让很多学生很反感. 不管你喜不喜欢,现在需要你做的是,就是按照老师的要求,写一个程序,模拟老师的询问.当然,老师有时候需要更新某位同学的成绩. Input 本题目包含多组测试,请处理到文件结束. 在每个测试的第一行,有两个正整数 N 和 M ( 0<N<=200000,0<M<5000 ),分别代表学生的数目和操作的数目. 学生ID编号分

HDOJ-1754(简单线段树)

最近开始重新学习线段树,先从最简单的开始吧! I Hate It Time Limit: 9000/3000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submission(s): 46762    Accepted Submission(s): 18316 Problem Description 很多学校流行一种比较的习惯.老师们很喜欢询问,从某某到某某当中,分数最高的是多少.这让很多学生很反感. 不管

初识线段树

今天算是见到了线段树这玩意儿了,,, 这个线段树是一棵二叉树,树中的每一个结点表示了一个区间[a,b].每一个叶子节点表示了一个单位区间.对于每一个非叶结点所表示的结点[a,b],其左儿子表示的区间为[a,(a+b)/2],右儿子表示的区间为[(a+b)/2+1,b]. 这个线段树可以使用指针来表示,也可以使用静态全局数组模拟(大小一般要开成节点的四倍哟O(∩_∩)O~). 其题目单节点更新.成端更新(需要用到延迟标记).区间合并.扫描线. 其常见操作如下:建立线段树(build).更新操作(u

[poj2104]可持久化线段树入门题(主席树)

解题关键:离线求区间第k小,主席树的经典裸题: 对主席树的理解:主席树维护的是一段序列中某个数字出现的次数,所以需要预先离散化,最好使用vector的erase和unique函数,很方便:如果求整段序列的第k小,我们会想到离散化二分和线段树的做法, 而主席树只是保存了序列的前缀和,排序之后,对序列的前缀分别做线段树,具有差分的性质,因此可以求任意区间的第k小,如果主席树维护索引,只需要求出某个数字在主席树中的位置,即为sort之后v中的索引:若要求第k大,建树时反向排序即可 1 #include

【BZOJ4942】[Noi2017]整数 线段树+DFS(卡过)

[BZOJ4942][Noi2017]整数 题目描述去uoj 题解:如果只有加法,那么直接暴力即可...(因为1的数量最多nlogn个) 先考虑加法,比较显然的做法就是将A二进制分解成log位,然后依次更新这log位,如果最高位依然有进位,那么找到最高位后面的第一个0,将中间的所有1变成0,那个0变成1.这个显然要用到线段树,但是复杂度是nlog2n的,肯定过不去. 于是我在考场上yy了一下,这log位是连续的,我们每次都要花费log的时间去修改一个岂不是很浪费?我们可以先在线段树上找到这段区间

bzoj1798: [Ahoi2009]Seq 维护序列seq 线段树

题目传送门 这道题就是线段树 先传乘法标记再传加法 #include<cstdio> #include<cstring> #include<algorithm> #define LL long long using namespace std; const int M=400010; LL read(){ LL ans=0,f=1,c=getchar(); while(c<'0'||c>'9'){if(c=='-') f=-1; c=getchar();}

Vijos P1066 弱弱的战壕【多解,线段树,暴力,树状数组】

弱弱的战壕 描述 永恒和mx正在玩一个即时战略游戏,名字嘛~~~~~~恕本人记性不好,忘了-_-b. mx在他的基地附近建立了n个战壕,每个战壕都是一个独立的作战单位,射程可以达到无限(“mx不赢定了?!?”永恒[email protected][email protected]). 但是,战壕有一个弱点,就是只能攻击它的左下方,说白了就是横纵坐标都不大于它的点(mx:“我的战壕为什么这么菜”ToT).这样,永恒就可以从别的地方进攻摧毁战壕,从而消灭mx的部队. 战壕都有一个保护范围,同它的攻击

luogu 1712 区间(线段树+尺取法)

题意:给出n个区间,求选择一些区间,使得一个点被覆盖的次数超过m次,最小的花费.花费指的是选择的区间中最大长度减去最小长度. 坐标值这么大,n比较小,显然需要离散化,需要一个技巧,把区间转化为半开半闭区间,然后线段树的每一个节点表示一个半开半闭区间. 接着我们注意到需要求最小的花费,且这个花费只与选择的区间集合中的最大长度和最小长度有关. 这意味着如果最大长度和最小长度一定,我们显然是需要把中间长度的区间尽量的选择进去使答案不会变的更劣. 不妨把区间按长度排序,枚举每个最小长度区间,然后最大区间

【BZOJ】1382: [Baltic2001]Mars Maps (线段树+扫描线)

1382: [Baltic2001]Mars Maps Time Limit: 5 Sec  Memory Limit: 64 MB Description 给出N个矩形,N<=10000.其坐标不超过10^9.求其面积并 Input 先给出一个数字N,代表有N个矩形. 接下来N行,每行四个数,代表矩形的坐标. Output 输出面积并 Sample Input 2 10 10 20 20 15 15 25 30 Sample Output 225 本以为是傻逼题,没想到不容易啊- 线段树+扫描