zoukankan      html  css  js  c++  java
  • Codeforces 600E

    With $Dsu on tree$ we can answer queries of this type:

    How many vertices in the subtree of vertex $v$ has some property in $O (n log n)$ time (for all of the queries)?

    这题写的是轻重儿子(重链剖分)版本的 $Dsu on tree$

    具体流程如下:

    每次先递归计算轻儿子,再单独递归重儿子,计算完后轻儿子的一些信息需要删掉,但是重儿子的信息无需删除,如此出解,相当于是优化了暴力的多余部分

    每个节点会作为轻儿子被计算,重链剖分上垂直有 $log n$ 条链,故复杂度 $O (n log n)$

    代码

      1 #include <iostream>
      2 #include <cstdio>
      3 #include <cstring>
      4 
      5 using namespace std;
      6 
      7 typedef long long LL;
      8 
      9 const int MAXN = 1e05 + 10;
     10 const int MAXM = 1e05 + 10;
     11 const int MAXC = 1e05 + 10;
     12 
     13 struct LinkedForwardStar {
     14     int to;
     15 
     16     int next;
     17 } ;
     18 
     19 LinkedForwardStar Link[MAXM << 1];
     20 int Head[MAXN]= {0};
     21 int size = 0;
     22 
     23 void Insert (int u, int v) {
     24     Link[++ size].to = v;
     25     Link[size].next = Head[u];
     26 
     27     Head[u] = size;
     28 }
     29 
     30 int N;
     31 int colour[MAXN];
     32 
     33 int son[MAXN]= {0};
     34 int subsize[MAXN]= {0};
     35 void DFS (int root, int father) {
     36     son[root] = - 1;
     37     subsize[root] = 1;
     38     for (int i = Head[root]; i; i = Link[i].next) {
     39         int v = Link[i].to;
     40         if (v == father)
     41             continue;
     42         DFS (v, root);
     43         subsize[root] += subsize[v];
     44         if (son[root] == - 1 || subsize[v] > subsize[son[root]])
     45             son[root] = v;
     46     }
     47 }
     48 int vis[MAXN]= {0};
     49 int total[MAXC]= {0};
     50 int maxv = 0;
     51 LL sum = 0;
     52 void calc (int root, int father, int delta) { // 统计答案
     53     total[colour[root]] += delta;
     54     if (delta > 0 && total[colour[root]] >= maxv) {
     55         if (total[colour[root]] > maxv)
     56             sum = 0, maxv = total[colour[root]];
     57         sum += colour[root];
     58     }
     59     for (int i = Head[root]; i; i = Link[i].next) {
     60         int v = Link[i].to;
     61         if (v == father || vis[v])
     62             continue;
     63         calc (v, root, delta);
     64     }
     65 }
     66 LL answer[MAXN]= {0};
     67 void Solve (int root, int father, int type) { // type表示是不是重儿子信息
     68     for (int i = Head[root]; i; i = Link[i].next) {
     69         int v = Link[i].to;
     70         if (v == father || v == son[root])
     71             continue;
     72         Solve (v, root, 0);
     73     }
     74     if (~ son[root])
     75         Solve (son[root], root, 1), vis[son[root]] = 1;
     76     calc (root, father, 1);
     77     answer[root] = sum;
     78     if (~ son[root])
     79         vis[son[root]] = 0;
     80     if (! type) // 如果是轻儿子信息就需删除
     81         calc (root, father, - 1), maxv = sum = 0;
     82 }
     83 
     84 int getnum () {
     85     int num = 0;
     86     char ch = getchar ();
     87 
     88     while (! isdigit (ch))
     89         ch = getchar ();
     90     while (isdigit (ch))
     91         num = (num << 3) + (num << 1) + ch - '0', ch = getchar ();
     92 
     93     return num;
     94 }
     95 
     96 int main () {
     97     N = getnum ();
     98     for (int i = 1; i <= N; i ++)
     99         colour[i] = getnum ();
    100     for (int i = 1; i < N; i ++) {
    101         int u = getnum (), v = getnum ();
    102         Insert (u, v), Insert (v, u);
    103     }
    104     DFS (1, 0), Solve (1, 0, 0);
    105     for (int i = 1; i <= N; i ++) {
    106         if (i > 1)
    107             putchar (' ');
    108         printf ("%lld", answer[i]);
    109     }
    110     puts ("");
    111 
    112     return 0;
    113 }
    114 
    115 /*
    116 4
    117 1 2 3 4
    118 1 2
    119 2 3
    120 2 4
    121 */
    122 
    123 /*
    124 15
    125 1 2 3 1 2 3 3 1 1 3 2 2 1 2 3
    126 1 2
    127 1 3
    128 1 4
    129 1 14
    130 1 15
    131 2 5
    132 2 6
    133 2 7
    134 3 8
    135 3 9
    136 3 10
    137 4 11
    138 4 12
    139 4 13
    140 */
  • 相关阅读:
    【云速建站】表单应用
    【云速建站】页面产品维护简述
    Forrester:华为云容器是容器混合云最佳选择
    AI如何驱动软件开发?华为云DevCloud 权威专家邀你探讨
    完美数据迁移-MongoDB Stream的应用
    补习系列(3)-springboot中的几种scope
    补习系列(2)-springboot mime类型处理
    关于Python的随机数模块,你必须要掌握!
    【Java】实战Java虚拟机之五“开启JIT编译”
    Jetty实战之 嵌入式Jetty运行Servlet
  • 原文地址:https://www.cnblogs.com/Colythme/p/10248071.html
Copyright © 2011-2022 走看看