zoukankan      html  css  js  c++  java
  • HDU 5044 TREE

    题意:一棵树上两种操作,操作1,改变u到v的每一点的值增加k,操作2,改变u到v每一条边值增加k。最后结束时问,每一点和每一条边的值。

    初始时,点和边的值都为0.

    分析:

    很显然要用树链剖分,将点和边分别划分成连续一段的编号,然后就是维护一段一段的值了,给它增加一个值,由于题目只需要输出最后结果,那么可以用树桩数组维护。

    以边为例,对于l~r的每个值增加k,利用树桩数组v[i]表示第i个值与前一个值的差值,那么第k条边的值就是sum{v[i]| 1 <= i <= k},更新时则只要给v[l]加k,给v[r+1]加(- k), 这个都可以累加下来,最后依次对每个v[i]更新,然后求出结果就行了。

    代码:

      1 #include <iostream>
      2 #include <algorithm>
      3 #include <vector>
      4 #include <map>
      5 #include <cstring>
      6 #include <set>
      7 #include <bitset>
      8 #include <cstdio>
      9 #include <cmath>
     10 #define esp 1e-8
     11 #pragma comment(linker, "/STACK:102400000,102400000")
     12 #define in freopen("F:\rootial\data\data.txt", "r", stdin);
     13 #define IN freopen("solve_in.txt", "r", stdin);
     14 #define out freopen("out.txt", "w", stdout);
     15 #define pf(x) ((x)*(x))
     16 #define lowbit(x) ((x)&(-(x)))
     17 #define bug(x) printf("Line %d: >>>>>>
    ", (x));
     18 #define lson l, m, rt<<1
     19 #define rson m+1, r, rt<<1|1
     20 #define pb push_back
     21 #define mp make_pair
     22 #define pi acos(-1.0)
     23 #define pf(x) ((x)*(x))
     24 
     25 using namespace std;
     26 typedef long long LL;
     27 typedef pair<int, int> PII;
     28 typedef map<LL,  LL> MPS;
     29 typedef MPS::iterator IT;
     30 
     31 const int maxn = (int)1e5 + 100;
     32 LL vv[2][maxn];
     33 int n, m;
     34 int top[maxn], num[maxn], tnum[maxn], ID[maxn], son[maxn], sz[maxn], fa[maxn], dep[maxn], tmp[maxn];
     35 int cnt;
     36 
     37 struct Edge
     38 {
     39     int v, id;
     40     Edge() {}
     41     Edge(int v, int id):v(v), id(id) {}
     42 } edge[maxn*2];
     43 int fi[maxn], nxt[maxn*2];
     44 LL q[2][maxn];
     45 
     46 void update(int x, LL v[], LL val)
     47 {
     48     while(x <= n)
     49     {
     50         v[x] +=val;
     51         x += lowbit(x);
     52     }
     53 }
     54 void update(int l, int r, LL val, LL v[])
     55 {
     56     update(l, v, val);
     57     update(r+1, v, -val);
     58 }
     59 
     60 void query(int x, LL &res1, LL &res2)
     61 {
     62     res1 = 0, res2 = 0;
     63     while(x > 0)
     64     {
     65         res1 += vv[0][x];
     66         res2 += vv[1][x];
     67         x -= lowbit(x);
     68     }
     69 }
     70 void dfs1(int u, int f)
     71 {
     72     fa[u] = f;
     73     dep[u] = dep[f] + 1;
     74     sz[u] = 1;
     75     for(int i = fi[u]; i; i = nxt[i])
     76     {
     77         int v = edge[i].v;
     78         int id = edge[i].id;
     79         if(v == f)
     80             continue;
     81         tmp[v] = id;
     82         dfs1(v, u);
     83         sz[u] += sz[v];
     84         if(sz[son[u]] < sz[v])
     85             son[u] = v;
     86     }
     87 }
     88 void dfs2(int u, int f)
     89 {
     90     if(son[u])
     91     {
     92         num[son[u]] = ++cnt;
     93         top[son[u]] = top[u];
     94         ID[tmp[son[u]]] = cnt;
     95         dfs2(son[u], u);
     96     }
     97     for(int i = fi[u]; i; i = nxt[i])
     98     {
     99         int v = edge[i].v;
    100         int id = edge[i].id;
    101         if(v == f || v == son[u]) continue;
    102         top[v] = v;
    103         num[v] = ++cnt;
    104         ID[tmp[v]] = cnt;
    105         dfs2(v, u);
    106     }
    107 }
    108 void init()
    109 {
    110     for(int i = 0; i <= n; i++)
    111     {
    112         vv[0][i] = vv[1][i] = 0;
    113         son[i] = 0;
    114         sz[i] = 0;
    115         fa[i] = 0;
    116         cnt = 0;
    117         fi[i] = 0;
    118         nxt[i<<1] = nxt[i<<1|1] = 0;
    119         q[0][i] = q[1][i] = 0;
    120     }
    121 }
    122 void add(int u, int v, int x)
    123 {
    124     edge[++cnt] = Edge(v, x);
    125     nxt[cnt] = fi[u];
    126     fi[u] = cnt;
    127     edge[++cnt] = Edge(u, x);
    128     nxt[cnt] = fi[v];
    129     fi[v] = cnt;
    130 }
    131 void input()
    132 {
    133     scanf("%d%d", &n, &m);
    134     init();
    135     for(int i = 1; i < n; i++)
    136     {
    137         int u, v;
    138         scanf("%d%d", &u, &v);
    139         add(u, v, i);
    140     }
    141     cnt = 0;
    142 }
    143 inline bool isdigit(char ch)
    144 {
    145     return ch >= '0' && ch <= '9';
    146 }
    147 const LL B = (int)1e5 + 10;
    148 
    149 void update(int u, int v, int k) //dian
    150 {
    151     while(top[u] != top[v])
    152     {
    153         if(dep[top[u]] < dep[top[v]]) swap(u, v);
    154         q[0][num[top[u]]] += k;
    155         q[0][num[u]+1] -= k;
    156         u = fa[top[u]];
    157     }
    158     if(dep[u] < dep[v]) swap(u, v);
    159     q[0][num[v]] += k;
    160     q[0][num[u]+1] -= k;
    161 }
    162 void update1(int u, int v, int k) //bian
    163 {
    164     while(top[u] != top[v])
    165     {
    166         if(dep[top[u]] < dep[top[v]]) swap(u, v);
    167         if(u != top[u])
    168         {
    169             q[1][num[son[top[u]]]] += k;
    170             q[1][num[u]+1] -= k;
    171         }
    172         u = top[u];
    173         q[1][num[u]] += k;
    174         q[1][num[u]+1] -= k;;
    175         u = fa[u];
    176     }
    177     if(dep[u] < dep[v]) swap(u, v);
    178     if(u != v)
    179     {
    180         q[1][num[son[v]]] += k;
    181         q[1][num[u]+1] -=  k;
    182     }
    183 }
    184 LL ans[maxn];
    185 
    186 void solve()
    187 {
    188     dfs1(1, 0);
    189     top[1] = 1;
    190     num[1] = ++cnt;
    191     dfs2(1, 0);
    192     for(int i = 0; i < m; i++)
    193     {
    194         char s[20];
    195         int u, v, k;
    196         scanf("%s%d%d%d", s, &u, &v, &k);
    197         if(strcmp(s, "ADD1") == 0)
    198         {
    199             update(u, v, k);
    200         }
    201         else
    202         {
    203             update1(u, v, k);
    204         }
    205     }
    206     for(int k = 0; k < 2; k++){
    207             for(int i = 1; i <= n; i++)
    208                 update(i, vv[k], q[k][i]);
    209         }
    210 
    211     for(int i = 1; i <= n; i++)
    212     {
    213         LL res1, res2;
    214         query(num[i], res1, res2);
    215         ans[num[i]] = res2;
    216         printf("%I64d%c", res1, i == n ? '
    ' : ' ');
    217     }
    218     for(int i = 1; i <= n-1; i++)
    219     {
    220         LL res = ans[ID[i]];
    221         printf("%I64d%c", res, i == n-1 ? '
    ' : ' ');
    222     }
    223     if(n == 1)
    224         puts("");
    225 }
    226 int main()
    227 {
    228 
    229     int T;
    230     for(int t = scanf("%d", &T); t <= T; t++)
    231     {
    232         printf("Case #%d:
    ", t);
    233         input();
    234         solve();
    235     }
    236     return 0;
    237 }
    View Code
  • 相关阅读:
    PHP7 快速编译安装
    php访问url的四种方式
    php获取文件 return array数组的值
    thinkphp3.2自定义success及error跳转页面
    think php 访问时
    vtk点云数据的显示[转]
    strlen() 和 sizeof() 在字符串中的使用
    函数 MultiByteToWideChar() 详解
    函数WideCharToMultiByte() 详解
    wchar_t 和 char 之间转换
  • 原文地址:https://www.cnblogs.com/rootial/p/3997045.html
Copyright © 2011-2022 走看看