HDU 4027 Can you answer these queries? 线段树裸题

题意:

给定2个操作

0、把区间的每个数sqrt

2、求和

因为每个数的sqrt次数很少,所以直接更新到底,用个标记表示是否更新完全(即区间内的数字只有0,1就不用再更新了)

#include<stdio.h>
#include<iostream>
#include<algorithm>
#include<vector>
#include<cmath>
#include<queue>
#include<set>
#include<map>
using namespace std;
#define N 1000000
#define L(x) (x<<1)
#define R(x) (x<<1|1)
#define Val(x) tree[x].val
#define Ned(x) tree[x].ned
#define ll __int64
inline ll Mid(ll x,ll y){return (x+y)>>1;}
struct node{
	ll l, r;
	ll val;//这个区间需要被sqrt几次
	bool ned;//如果为true则表示这个区间不管怎么开结果都一样
}tree[N*4];
ll n, a[101000];
void push_up(ll id){
	Ned(id) = Ned(L(id))&&Ned(R(id));
	Val(id) = Val(L(id))+Val(R(id));
}
void build(ll l, ll r, ll id){
	tree[id].l = l; tree[id].r = r;
	Ned(id) = false;
	if(l==r){ Val(id) = a[l]; if(Val(id)<=1)Ned(id)=true; return ;}
	ll mid = Mid(l,r);
	build(l, mid, L(id));
	build(mid+1,r,R(id));
	push_up(id);
}
void update(ll l, ll r, ll id){
	if(Ned(id))return;
	if(l == tree[id].l && tree[id].r == r && l==r){
		Val(id) = (ll)sqrt(1.0*Val(id));
		if(Val(id)<=1)Ned(id)=1;
		return;
	}
	ll mid = Mid(tree[id].l, tree[id].r);
	if(r<=mid)
		update(l,r,L(id));
	else if(mid<l)
		update(l,r,R(id));
	else
	{
		update(l,mid,L(id));
		update(mid+1,r,R(id));
	}
	push_up(id);
}
ll query(ll l, ll r, ll id){
	if(tree[id].l == tree[id].r)return Val(id);
	if(l == tree[id].l && tree[id].r == r && Ned(id))return Val(id);
	ll mid = Mid(tree[id].l, tree[id].r);
	if(r<=mid)return query(l,r,L(id));
	else if(mid<l)return query(l,r,R(id));
	else return query(l,mid,L(id))+query(mid+1,r,R(id));
}
int main(){
	ll u, v, i, que, Cas = 1;
	while(~scanf("%I64d",&n)){
		for(i=1;i<=n;i++)scanf("%I64d",&a[i]);
		build(1,n,1);
		scanf("%I64d",&que);
		printf("Case #%I64d:\n",Cas++);
		while(que--){
			scanf("%I64d %I64d %I64d",&i,&u,&v);	if(u>v)swap(u,v);
			if(i==0)
				update(u,v,1);
			else
				printf("%I64d\n",query(u,v,1));
		}
		puts("");
	}
	return 0;
}

/*
10
1 2 3 4 5 6 7 8 9 10
99
1 10 10
0 2 8

*/

HDU 4027 Can you answer these queries? 线段树裸题,布布扣,bubuko.com

时间: 2024-10-22 09:52:08

HDU 4027 Can you answer these queries? 线段树裸题的相关文章

HDU 4027 Can you answer these queries? (线段树+区间点修改)

题意:给你 n 个数,m个询问(c,x,y) c==0 把x,y区间的值变为原来的平方根(向下取整) c==1 计算x,y区间的和. 利用1的开方永远为1剪枝.. #include<cstdio> #include<stdlib.h> #include<string.h> #include<string> //#include<map> #include<cmath> #include<iostream> #include

HDU 4027 Can you answer these queries?(线段树,区间更新,区间查询)

题目 线段树 简单题意: 区间(单点?)更新,区间求和 更新是区间内的数开根号并向下取整 这道题不用延迟操作 //注意: //1:查询时的区间端点可能前面的比后面的大: //2:优化:因为每次更新都是开平方,同一个数更新有限次数就一直是1了,所以可以这样优化 #include <stdio.h> #include<math.h> #define N 100010 #define LL __int64 #define lson l,m,rt<<1 #define rson

hdu 4027 Can you answer these queries? 线段树区间开根号,区间求和

Can you answer these queries? Time Limit: 1 Sec  Memory Limit: 256 MB 题目连接 http://acm.hdu.edu.cn/showproblem.php?pid=5195 Description A lot of battleships of evil are arranged in a line before the battle. Our commander decides to use our secret weapo

HDU 4027 Can you answer these queries?(线段树)

Can you answer these queries? Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 65768/65768 K (Java/Others)Total Submission(s): 9216    Accepted Submission(s): 2106 Problem Description A lot of battleships of evil are arranged in a line before

HDU 4027 Can you answer these queries?(线段树 区间不等更新)

题意  输入n个数  然后有两种操作   输入0时将给定区间所有数都变为自己的开方   输入1输出给定区间所有数的和 虽然是区间更新  但每个点更新的不一样  因此只能对单点进行更新  其实一个点最多被更新7次  2^64开平方7次后就变为1了  如果某个区间的数都变为了1  那么对这个区间的开方就不用考虑了   另外要注意给你的区间可能是反的 #include <bits/stdc++.h> #define lc p<<1,s,mid #define rc p<<1|

HDU 4027 Can you answer these queries? 线段树,区间修改

Can you answer these queries? Problem Description A lot of battleships of evil are arranged in a line before the battle. Our commander decides to use our secret weapon to eliminate the battleships. Each of the battleships can be marked a value of end

HDU - 4027 Can you answer these queries? (线段树区间更新+思维)

题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4027 题意:给定两种操作,查询(求出区间内的和),更新(区间内每个值都开根号,取整数) 题目给出所有数字之和小于263,所以最大的数最多7次也就变成1.所以在更新的时候加个判断条件,提前结束. 然后这道题目还有个坑,L有可能比R大,需要交换一下,(╬▔皿▔) . 1 #include <cstdio> 2 #include <algorithm> 3 #include <cmat

HDU 4027 Can you answer these queries?(线段树的单点更新+区间查询)

题目链接 题意 : 给你N个数,进行M次操作,0操作是将区间内的每一个数变成自己的平方根(整数),1操作是求区间和. 思路 :单点更新,区间查询,就是要注意在更新的时候要优化,要不然会超时,因为所有的数开几次方之后都会变成1,所以到了1不用没完没了的更新. 1 //HDU 4027 2 #include <cstdio> 3 #include <cstring> 4 #include <cmath> 5 #include <iostream> 6 #defi

HDU4027 Can you answer these queries 线段树区间求和+剪枝

给了你n,然后n个数字在一个数组中,接下来m个询问,每个询问三个数字 t,x,y,若t==0,那么修改区间[x,y]的每一个值,变为原来每个位置上的数 开根号取整,若t==1,那么对区间[x,y]求和 由于n,m,很大,所以树状数组铁定超时,若直接用线段树来做区间修改,那么也是超时,这类题目没别的方法了,静心剪枝,发现题目给的数据范围为2^63,有没有发现,2^63开根号 绝对不需要开10次,就能到1,到1以后就不需要再开了,意思就是若有某个区间[x,y]每一个点的值都为1时,这一段区间事实上是