zoukankan      html  css  js  c++  java
  • POJ 3321 Apple Tree (树状数组+dfs序)

    题目链接:http://poj.org/problem?id=3321

    给你n个点,n-1条边,1为根节点。给你m条操作,C操作是将x点变反(1变0,0变1),Q操作是询问x节点以及它子树的值之和。初始所有的节点为1。

    用DFS序的方法将以1为根节点DFS遍历所有的节点,L[i]表示i点出现的最早的时间戳,R[i]表示i点出现的最晚的时间戳,每个节点出现两次。

    所以要是查询 i 及它子树的值的和之话,只要用树状数组查询L[i]~R[i]之间的值然后除以2,复杂度log(n)。改变操作的话,只要改变下标为L[i]以及R[i]上的值就可以了。

    类似的题目有HDU3974,不过是线段树成段更新,也是用DFS序的方法写的。

     1 #include <iostream>
     2 #include <cstring>
     3 #include <cstdio>
     4 #include <vector>
     5 using namespace std;
     6 const int MAXN = 2e5 + 5;
     7 const int INF = 1e9;
     8 int head[MAXN] , n , m , cnt , L[MAXN] , R[MAXN] , bit[MAXN * 2] , dfn , a[MAXN];
     9 //a数组表示每个元素的值
    10 struct data {
    11     int next , to;
    12 }edge[MAXN * 2];
    13 
    14 inline void add_edge(int u , int v) {
    15     edge[cnt].next = head[u];
    16     edge[cnt].to = v;
    17     head[u] = cnt++;
    18 }
    19 
    20 void dfs(int u , int par) {
    21     L[u] = ++dfn;
    22     for(int i = head[u] ; ~i ; i = edge[i].next) {
    23         int v = edge[i].to;
    24         if(v == par)
    25             continue;
    26         dfs(v , u);
    27     }
    28     R[u] = ++dfn;
    29 }
    30 
    31 inline void add(int i , int x) {
    32     for( ; i <= dfn ; i += (i & -i))
    33         bit[i] += x;
    34 }
    35 
    36 int sum(int i) {
    37     int s = 0;
    38     for( ; i >= 1 ; i -= (i & -i))
    39         s += bit[i];
    40     return s;
    41 }
    42 
    43 int main()
    44 {
    45     int u , v;
    46     char q[5];
    47     while(~scanf("%d" , &n)) {
    48         memset(head , -1 , sizeof(head));
    49         memset(bit , 0 , sizeof(bit));
    50         memset(a , 0 , sizeof(a));
    51         cnt = dfn = 0;
    52         for(int i = 1 ; i < n ; i++) {
    53             scanf("%d %d" , &u , &v);
    54             add_edge(u , v);
    55             add_edge(v , u);
    56         }
    57         dfs(1 , -1);
    58         scanf("%d" , &m);
    59         while(m--) {
    60             scanf("%s %d" , q , &u);
    61             if(q[0] == 'Q') {
    62                 printf("%d
    " , ((R[u] - L[u] + 1) - (sum(R[u]) - sum(L[u] - 1))) / 2); //除2因为dfs序中 每个点出现2次
    63             }
    64             else {
    65                 if(a[u]) {
    66                     add(L[u] , -1);
    67                     add(R[u] , -1);
    68                     a[u] = 0;
    69                 }
    70                 else {
    71                     add(L[u] , 1);
    72                     add(R[u] , 1);
    73                     a[u] = 1;
    74                 }
    75             }
    76         }
    77     }
    78 }
  • 相关阅读:
    2020软件工程第四次作业04
    2020软件工程作业02
    2020软件工程作业01
    2020软件工程个人作业06——软件工程实践总结作业
    2020软件工程作业05
    2020软件工程作业00—问题清单
    2020软件工程作业03
    2020软件工程作业02
    2020软件工程作业01
    小小小-冲刺集合
  • 原文地址:https://www.cnblogs.com/Recoder/p/5405326.html
Copyright © 2011-2022 走看看