首页 > 其他 > 详细

Partition(线段树的离线处理)

时间:2014-06-02 18:20:26      阅读:470      评论:0      收藏:0      [点我收藏+]

有一点类似区间K值的求法。

这里有两颗树,一个是自己建的线段树,一个是题目中给定的树。以线段树和树进行区分。

首先离散化一下,以离散化后的结果建线段树,线段树的节点开了2维,一维保存当前以当前节点为权值的树的节点是往左走的,另一维是往右走的,用一个vector保存一下以当前i节点为结束的询问,因为所有的询问都是从根节点出发的,只需保存终点即可。

然后从根节点出发遍历整棵树,当走到当前节点时,枚举以当前节点的询问q[i],然后求出比q[i].x大的向左和向右走的节点个数,以及比它小的个数,如果有相等的直接为0,最后根据可能性加起来即可。

每走完一个节点,回退时,把当前节点更新为未走。

把树节点的权值跟询问的搞混了,WA一次。。搞了组数据才看出来

bubuko.com,布布扣
  1 #include <iostream>
  2 #include<cstdio>
  3 #include<cstring>
  4 #include<algorithm>
  5 #include<stdlib.h>
  6 #include<vector>
  7 #include<cmath>
  8 #include<queue>
  9 #include<set>
 10 #include<map>
 11 using namespace std;
 12 #define N 100010
 13 #define LL long long
 14 #define INF 0xfffffff
 15 const double eps = 1e-8;
 16 const double pi = acos(-1.0);
 17 const double inf = ~0u>>2;
 18 #pragma comment(linker, "/STACK:1024000000,1024000000")
 19 int s[N<<2][2],a[N<<1],b[N];
 20 map<int,int>f;
 21 vector<int>ed[N];//儿子
 22 vector<int>dd[N];//以i节点结束的询问
 23 int g ;
 24 struct node{
 25     int v,x;
 26     int ax,ay,f;
 27 }p[N];
 28 void up(int w)
 29 {
 30     s[w][0] = s[w<<1][0]+s[w<<1|1][0];//向左走
 31     s[w][1] = s[w<<1][1]+s[w<<1|1][1];//向右走
 32 }
 33 void build(int l,int r,int w)
 34 {
 35     if(l==r)
 36     {
 37         s[w][0] = s[w][1] = 0;
 38         return ;
 39     }
 40     int m = (l+r)>>1;
 41     build(l,m,w<<1);
 42     build(m+1,r,w<<1|1);
 43     up(w);
 44 }
 45 void update(int p,int d,int dir,int l,int r,int w)
 46 {
 47     if(l==r)
 48     {
 49         s[w][dir] += d;
 50         return ;
 51     }
 52     int m = (l+r)>>1;
 53     if(p<=m) update(p,d,dir,l,m,w<<1);
 54     else update(p,d,dir,m+1,r,w<<1|1);
 55     up(w);
 56 }
 57 int query(int a,int b,int dir,int l,int r,int w)
 58 {
 59     if(a<=l&&b>=r)
 60     {
 61         return s[w][dir];
 62     }
 63     int m = (l+r)>>1;
 64     int res=0;
 65     if(a<=m) res+=query(a,b,dir,l,m,w<<1);
 66     if(b>m) res+=query(a,b,dir,m+1,r,w<<1|1);
 67     return res;
 68 }
 69 void dfs(int u,int pre)
 70 {
 71     int i;
 72     for(i = 0; i < dd[u].size(); i++)
 73     {
 74         int k = dd[u][i];
 75         int id = f[p[k].x];
 76        // cout<<k<<endl;
 77         int cnt1 = query(id,id,0,1,g,1);
 78         int cnt2 = query(id,id,1,1,g,1);
 79         //cout<<cnt1<<" "<<cnt2<<" "<<u<<" "<<id<<endl;
 80         if(cnt1||cnt2)
 81         {
 82             p[k].f = 0;
 83             continue;
 84         }
 85         else
 86         {
 87             p[k].f = 1;
 88             int l0 = query(1,id,0,1,g,1);
 89             int l1 = query(1,id,1,1,g,1);
 90             int r0 = query(id,g,0,1,g,1);
 91             int r1 = query(id,g,1,1,g,1);
 92            // cout<<l0<<" "<<l1<<" "<<r0<<" "<<r1<<endl;
 93             p[k].ay = r0+r1+3*(l1+l0);
 94             p[k].ax = l1;
 95         }
 96     }
 97     for(i = 0;i < ed[u].size() ; i++)
 98     {
 99         int v = ed[u][i];
100         int id = f[b[u]];
101        // cout<<id<<" "<<u<<endl;
102         if(i==0)
103         {
104             update(id,1,0,1,g,1);
105         }
106         else update(id,1,1,1,g,1);
107         dfs(v,u);
108         if(i==0)
109         update(id,-1,0,1,g,1);
110         else update(id,-1,1,1,g,1);
111     }
112 }
113 int main()
114 {
115     int t,i,n,m,q;
116     cin>>t;
117     while(t--)
118     {
119         scanf("%d",&n);
120         f.clear();
121         g = 0;
122         for(i = 1; i <=n; i++)
123         {
124             scanf("%d",&a[i]);
125             b[i] = a[i];
126             ed[i].clear();
127             dd[i].clear();
128         }
129         scanf("%d",&m);
130         for(i = 1; i <= m; i++)
131         {
132             int u,l,r;
133             scanf("%d%d%d",&u,&l,&r);
134             ed[u].push_back(l);
135             ed[u].push_back(r);
136         }
137         scanf("%d",&q);
138         for(i = 1;i <= q ;i++)
139         {
140             scanf("%d%d",&p[i].v,&p[i].x);
141             dd[p[i].v].push_back(i);
142             a[n+i] = p[i].x;
143         }
144         sort(a+1,a+n+q+1);
145         f[a[1]] = ++g;
146         for(i = 2; i <= n+q ; i++)
147         {
148             if(a[i]!=a[i-1])
149             f[a[i]] = ++g;
150         }
151         build(1,g,1);
152         dfs(1,-1);
153         for(i = 1; i <= q;  i++)
154         {
155             if(p[i].f==0)
156             printf("%d\n",0);
157             else
158             printf("%d %d\n",p[i].ax,p[i].ay);
159         }
160     }
161     return 0;
162 }
View Code

 

 

Partition(线段树的离线处理),布布扣,bubuko.com

Partition(线段树的离线处理)

原文:http://www.cnblogs.com/shangyu/p/3764402.html

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!