<题目链接>

题目大意:

给定一段序列,现在对指定区间进行两种操作:一是对指定区间进行修改,对其中的每个数字都开根号(开根号后的数字仍然取整);二是对指定区间进行查询,查询这段区间所有数字的和。

解题分析:

本题虽然是区间修改,但是不需要用 lazy标记,因为要对指定区间的每个数进行开根号的处理,也就是说,每次 update ,都要延伸到该区间涉及到的叶子节点,进行开根,而不是在叶子节点上端的某个节点就将开根的指令存储下来。那么是不是说我们每次只能对 update 的每个区间所涉及到的每个节点进行暴力的单点修改呢?很显然不是的,因为每个节点的值不超过2^63,所以每个值的有效开方次数并不多。所以我们对线段树的每个节点引入一个标记cnt,用它来记录该节点对应的区域是否全部不需要开方,如果不需要开方,那么就直接return ,终止无效更新,从而提高效率。

 1 #include <cstdio>
 2 #include <cstring>
 3 #include <cmath>
 4 #include <algorithm>
 5 using namespace std;
 6 
 7 #define Lson rt<<1,l,mid
 8 #define Rson rt<<1|1,mid+1,r
 9 typedef long long ll;
10 const int M = 1e5+5;
11 int n;
12 ll tr[M<<2],arr[M];
13 bool fp[M<<2];
14 void Pushup(int rt){
15     if(fp[rt<<1]&&fp[rt<<1|1])fp[rt]=true;    //如果两个子区间全部不用开根号的话,那么该区间也标记为不用继续开根
16     else fp[rt]=false;
17     tr[rt]=tr[rt<<1]+tr[rt<<1|1];
18 }
19 void build(int rt,int l,int r){
20     if(l==r){
21         tr[rt]=arr[l];
22         return;
23     }
24     int mid=(l+r)>>1;
25     build(Lson);
26     build(Rson);
27     Pushup(rt);
28 }
29 void update(int rt,int l,int r,int L,int R){
30     if(fp[rt])return;      //如果遍历到不用继续向下更新的区间,则直接返回
31     if(l==r){
32         tr[rt]=sqrt(tr[rt]*1.0);
33         if(tr[rt]==1)fp[rt]=true;    //如果tr[rt]==1,那么该点就标记为不用继续开根
34         return;
35     }
36     int mid=(l+r)>>1;
37     if(L<=mid)update(Lson,L,R);
38     if(R>mid)update(Rson,L,R);
39     Pushup(rt);
40 }
41 ll query(int rt,int l,int r,int L,int R){
42     if(L<=l&&r<=R){
43         return tr[rt];
44     }
45     int mid=(l+r)>>1;
46     ll ans=0;
47     if(L<=mid)ans+=query(Lson,L,R);
48     if(R>mid)ans+=query(Rson,L,R);
49     return ans;
50 }
51 int main(){
52     int ncase=0;
53     while(scanf("%d",&n)!=EOF){
54         memset(fp,false,sizeof(fp));
55         for(int i=1;i<=n;i++)scanf("%lld",&arr[i]);
56         build(1,1,n);
57         int m;scanf("%d",&m);
58         printf("Case #%d:\n",++ncase);
59         while(m--){
60             int op,x,y;
61             scanf("%d%d%d",&op,&x,&y);
62             if(x>y)swap(x,y);    //注意这里,坑
63             if(op==0){
64                 update(1,1,n,x,y);
65             }
66             else{
67                 printf("%lld\n",query(1,1,n,x,y));
68             }
69         }
70         printf("\n");
71     }
72     return 0;
73 }

 

 

2018-09-23